mirror of
https://github.com/grocy/grocy.git
synced 2025-04-28 17:23:56 +00:00
Make it possible to actively not-check a mandatory checkbox Userfield (closes #2601) Pluralize the "opened" localization string (closes #2280) Added a trendline to the price history chart (closes #2237) Various minor style/code refinements
430 lines
14 KiB
JavaScript
430 lines
14 KiB
JavaScript
Grocy.Components.ProductPicker = {};
|
|
|
|
Grocy.Components.ProductPicker.GetPicker = function()
|
|
{
|
|
return $('#product_id');
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.GetInputElement = function()
|
|
{
|
|
return $('#product_id_text_input');
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.GetValue = function()
|
|
{
|
|
return $('#product_id').val();
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.SetValue = function(value)
|
|
{
|
|
Grocy.Components.ProductPicker.GetInputElement().val(value);
|
|
Grocy.Components.ProductPicker.GetInputElement().trigger('change');
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.SetId = function(value)
|
|
{
|
|
Grocy.Components.ProductPicker.GetPicker().val(value);
|
|
Grocy.Components.ProductPicker.GetPicker().data('combobox').refresh();
|
|
Grocy.Components.ProductPicker.GetInputElement().trigger('change');
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.Clear = function()
|
|
{
|
|
Grocy.Components.ProductPicker.SetValue('');
|
|
Grocy.Components.ProductPicker.SetId(null);
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.InProductAddWorkflow = function()
|
|
{
|
|
return GetUriParam('flow') == "InplaceNewProductWithName";
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.InProductModifyWorkflow = function()
|
|
{
|
|
return GetUriParam('flow') == "InplaceAddBarcodeToExistingProduct";
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.InAnyFlow = function()
|
|
{
|
|
return Grocy.Components.ProductPicker.InProductAddWorkflow() || Grocy.Components.ProductPicker.InProductModifyWorkflow();
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.FinishFlow = function()
|
|
{
|
|
if (GetUriParam("flow") == "InplaceAddBarcodeToExistingProduct")
|
|
{
|
|
$("#product_id option[value=\"" + Grocy.Components.ProductPicker.GetValue() + "\"]").attr("data-additional-searchdata", (i, value) => `${value || ""}${GetUriParam("barcode")},`);
|
|
}
|
|
|
|
RemoveUriParam("flow");
|
|
RemoveUriParam("barcode");
|
|
RemoveUriParam("product-name");
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.ShowCustomError = function(text)
|
|
{
|
|
var element = $("#custom-productpicker-error");
|
|
element.text(text);
|
|
element.removeClass("d-none");
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.HideCustomError = function()
|
|
{
|
|
$("#custom-productpicker-error").addClass("d-none");
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.Disable = function()
|
|
{
|
|
Grocy.Components.ProductPicker.GetInputElement().attr("disabled", "");
|
|
$("#camerabarcodescanner-start-button").attr("disabled", "");
|
|
$("#camerabarcodescanner-start-button").addClass("disabled");
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.Enable = function()
|
|
{
|
|
Grocy.Components.ProductPicker.GetInputElement().removeAttr("disabled");
|
|
$("#camerabarcodescanner-start-button").removeAttr("disabled");
|
|
$("#camerabarcodescanner-start-button").removeClass("disabled");
|
|
}
|
|
|
|
$('.product-combobox').combobox({
|
|
appendId: '_text_input',
|
|
bsVersion: '4',
|
|
clearIfNoMatch: false
|
|
});
|
|
|
|
var prefillProduct = GetUriParam('product-name');
|
|
var prefillProduct2 = Grocy.Components.ProductPicker.GetPicker().parent().data('prefill-by-name').toString();
|
|
if (prefillProduct2)
|
|
{
|
|
prefillProduct = prefillProduct2;
|
|
}
|
|
if (typeof prefillProduct !== "undefined")
|
|
{
|
|
var possibleOptionElement = $("#product_id option[data-additional-searchdata*=\"" + prefillProduct + "\"]").first();
|
|
if (possibleOptionElement.length === 0)
|
|
{
|
|
possibleOptionElement = $("#product_id option:contains(\"" + prefillProduct + "\")").first();
|
|
}
|
|
|
|
if (possibleOptionElement.length > 0)
|
|
{
|
|
$('#product_id').val(possibleOptionElement.val());
|
|
$('#product_id').data('combobox').refresh();
|
|
$('#product_id').trigger('change');
|
|
|
|
var nextInputElement = $(Grocy.Components.ProductPicker.GetPicker().parent().data('next-input-selector').toString());
|
|
nextInputElement.focus();
|
|
}
|
|
}
|
|
|
|
var prefillProductId = GetUriParam("product");
|
|
var prefillProductId2 = Grocy.Components.ProductPicker.GetPicker().parent().data('prefill-by-id').toString();
|
|
if (prefillProductId2)
|
|
{
|
|
prefillProductId = prefillProductId2;
|
|
}
|
|
if (typeof prefillProductId !== "undefined")
|
|
{
|
|
$('#product_id').val(prefillProductId);
|
|
|
|
if ($('#product_id').val() != null)
|
|
{
|
|
$('#product_id').data('combobox').refresh();
|
|
$('#product_id').trigger('change');
|
|
|
|
var nextInputElement = $(Grocy.Components.ProductPicker.GetPicker().parent().data('next-input-selector').toString());
|
|
nextInputElement.focus();
|
|
}
|
|
else
|
|
{
|
|
Grocy.Components.ProductPicker.GetInputElement().focus();
|
|
}
|
|
}
|
|
|
|
if (GetUriParam("flow") === "InplaceAddBarcodeToExistingProduct")
|
|
{
|
|
$('#InplaceAddBarcodeToExistingProduct').text(GetUriParam("barcode"));
|
|
$('#flow-info-InplaceAddBarcodeToExistingProduct').removeClass('d-none');
|
|
$('#barcode-lookup-disabled-hint').removeClass('d-none');
|
|
$('#barcode-lookup-hint').addClass('d-none');
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.PopupOpen = false;
|
|
$('#product_id_text_input').on('blur', function(e)
|
|
{
|
|
if (Grocy.Components.ProductPicker.GetPicker().hasClass("combobox-menu-visible"))
|
|
{
|
|
return;
|
|
}
|
|
$('#product_id').attr("barcode", "null");
|
|
|
|
var input = $('#product_id_text_input').val().toString();
|
|
var possibleOptionElement = [];
|
|
|
|
// Grocycode handling
|
|
if (input.startsWith("grcy"))
|
|
{
|
|
var gc = input.split(":");
|
|
if (gc[1] == "p")
|
|
{
|
|
possibleOptionElement = $("#product_id option[value=\"" + gc[2] + "\"]").first();
|
|
$("#product_id").data("grocycode", true);
|
|
$('#product_id').attr("barcode", input);
|
|
}
|
|
}
|
|
else // Normal product barcode handling
|
|
{
|
|
possibleOptionElement = $("#product_id option[data-additional-searchdata*=\"" + input.toLowerCase() + ",\"]").first();
|
|
}
|
|
|
|
if (GetUriParam('flow') === undefined && input.length > 0 && possibleOptionElement.length > 0)
|
|
{
|
|
$('#product_id').val(possibleOptionElement.val());
|
|
$('#product_id').attr("barcode", input);
|
|
$('#product_id').data('combobox').refresh();
|
|
$('#product_id').trigger('change');
|
|
}
|
|
else
|
|
{
|
|
if (Grocy.Components.ProductPicker.PopupOpen === true)
|
|
{
|
|
return;
|
|
}
|
|
|
|
var optionElement = $("#product_id option:contains(\"" + input + "\")").first();
|
|
if (input.length > 0 && optionElement.length === 0 && GetUriParam('flow') === undefined && Grocy.Components.ProductPicker.GetPicker().parent().data('disallow-all-product-workflows').toString() === "false")
|
|
{
|
|
var addProductWorkflowsAdditionalCssClasses = "";
|
|
if (Grocy.Components.ProductPicker.GetPicker().parent().data('disallow-add-product-workflows').toString() === "true")
|
|
{
|
|
addProductWorkflowsAdditionalCssClasses = "d-none";
|
|
}
|
|
|
|
var embedded = "";
|
|
if (GetUriParam("embedded") !== undefined)
|
|
{
|
|
embedded = "embedded";
|
|
}
|
|
|
|
var buttons = {
|
|
cancel: {
|
|
label: __t('Cancel'),
|
|
className: 'btn-secondary responsive-button',
|
|
callback: function()
|
|
{
|
|
Grocy.Components.ProductPicker.PopupOpen = false;
|
|
Grocy.Components.ProductPicker.SetValue('');
|
|
}
|
|
},
|
|
addnewproduct: {
|
|
label: '<strong>P</strong> ' + __t('Add as new product'),
|
|
className: 'btn-success add-new-product-dialog-button responsive-button ' + addProductWorkflowsAdditionalCssClasses,
|
|
callback: function()
|
|
{
|
|
// Not the best place here - this is only relevant when this flow is started from the shopping list item form
|
|
// (to select the correct shopping list on return)
|
|
if (GetUriParam("list") !== undefined)
|
|
{
|
|
embedded += "&list=" + GetUriParam("list");
|
|
}
|
|
|
|
Grocy.Components.ProductPicker.PopupOpen = false;
|
|
window.location.href = U('/product/new?flow=InplaceNewProductWithName&name=' + encodeURIComponent(input) + '&returnto=' + encodeURIComponent(Grocy.CurrentUrlRelative + "?flow=InplaceNewProductWithName&" + embedded) + "&" + embedded);
|
|
}
|
|
},
|
|
addbarcode: {
|
|
label: '<strong>B</strong> ' + __t('Add as barcode to existing product'),
|
|
className: 'btn-info add-new-barcode-dialog-button responsive-button',
|
|
callback: function()
|
|
{
|
|
Grocy.Components.ProductPicker.PopupOpen = false;
|
|
window.location.href = U(Grocy.CurrentUrlRelative + '?flow=InplaceAddBarcodeToExistingProduct&barcode=' + encodeURIComponent(input) + "&" + embedded);
|
|
}
|
|
},
|
|
addnewproductwithbarcode: {
|
|
label: '<strong>A</strong> ' + __t('Add as new product and prefill barcode'),
|
|
className: 'btn-warning add-new-product-with-barcode-dialog-button responsive-button ' + addProductWorkflowsAdditionalCssClasses,
|
|
callback: function()
|
|
{
|
|
Grocy.Components.ProductPicker.PopupOpen = false;
|
|
window.location.href = U('/product/new?flow=InplaceNewProductWithBarcode&barcode=' + encodeURIComponent(input) + '&returnto=' + encodeURIComponent(Grocy.CurrentUrlRelative + "?flow=InplaceAddBarcodeToExistingProduct&barcode=" + input + "&" + embedded) + "&" + embedded);
|
|
}
|
|
}
|
|
};
|
|
|
|
if (Grocy.ExternalBarcodeLookupPluginName)
|
|
{
|
|
buttons.barcodepluginlookup = {
|
|
label: '<strong>E</strong> ' + __t('External barcode lookup') + ' <span class="badge badge-pill badge-light">' + Grocy.ExternalBarcodeLookupPluginName + '</span>',
|
|
className: 'btn-dark add-new-product-plugin-dialog-button responsive-button ' + addProductWorkflowsAdditionalCssClasses,
|
|
callback: function()
|
|
{
|
|
Grocy.Components.ProductPicker.PopupOpen = false;
|
|
Grocy.FrontendHelpers.BeginUiBusy($("form").first().attr("id"));
|
|
|
|
Grocy.Api.Get("stock/barcodes/external-lookup/" + encodeURIComponent(input) + "?add=true",
|
|
function(pluginResponse)
|
|
{
|
|
if (pluginResponse == null)
|
|
{
|
|
toastr.warning(__t("Nothing was found for the given barcode"));
|
|
Grocy.FrontendHelpers.EndUiBusy($("form").first().attr("id"));
|
|
}
|
|
else
|
|
{
|
|
window.location.href = U("/product/" + pluginResponse.id + "?flow=InplaceNewProductByExternalBarcodeLookupPlugin&returnto=" + encodeURIComponent(Grocy.CurrentUrlRelative + "?flow=InplaceNewProductWithName&" + embedded) + "&" + embedded);
|
|
}
|
|
},
|
|
function(xhr)
|
|
{
|
|
Grocy.FrontendHelpers.ShowGenericError("Error while executing the barcode lookup plugin", xhr.response);
|
|
Grocy.FrontendHelpers.EndUiBusy($("form").first().attr("id"));
|
|
}
|
|
);
|
|
}
|
|
};
|
|
}
|
|
|
|
if (!Grocy.FeatureFlags.GROCY_FEATURE_FLAG_DISABLE_BROWSER_BARCODE_CAMERA_SCANNING)
|
|
{
|
|
buttons.retrycamerascanning = {
|
|
label: '<strong>C</strong> <i class="fa-solid fa-camera"></i>',
|
|
className: 'btn-primary responsive-button retry-camera-scanning-button',
|
|
callback: function()
|
|
{
|
|
Grocy.Components.ProductPicker.PopupOpen = false;
|
|
Grocy.Components.ProductPicker.SetValue('');
|
|
$("#camerabarcodescanner-start-button").click();
|
|
}
|
|
};
|
|
}
|
|
|
|
// The product picker contains only in-stock products on some pages,
|
|
// so only show the workflow dialog when the entered input
|
|
// does not match in existing product (name) or barcode,
|
|
// otherwise an error validation message that the product is not in stock
|
|
var existsAsProduct = false;
|
|
var existsAsBarcode = false;
|
|
Grocy.Api.Get('objects/product_barcodes_view?query[]=barcode=' + input,
|
|
function(barcodeResult)
|
|
{
|
|
if (barcodeResult.length > 0)
|
|
{
|
|
existsAsBarcode = true;
|
|
}
|
|
|
|
Grocy.Api.Get('objects/products?query[]=name=' + input,
|
|
function(productResult)
|
|
{
|
|
if (productResult.length > 0)
|
|
{
|
|
existsAsProduct = true;
|
|
}
|
|
|
|
if (!existsAsBarcode && !existsAsProduct)
|
|
{
|
|
Grocy.Components.ProductPicker.PopupOpen = true;
|
|
bootbox.dialog({
|
|
message: __t('"%s" could not be resolved to a product, how do you want to proceed?', input),
|
|
title: __t('Create or assign product'),
|
|
onEscape: function()
|
|
{
|
|
Grocy.Components.ProductPicker.PopupOpen = false;
|
|
Grocy.Components.ProductPicker.SetValue('');
|
|
},
|
|
size: 'large',
|
|
backdrop: true,
|
|
closeButton: false,
|
|
buttons: buttons,
|
|
className: "wider"
|
|
}).on('keypress', function(e)
|
|
{
|
|
if (e.key === 'B' || e.key === 'b')
|
|
{
|
|
$('.add-new-barcode-dialog-button').not(".d-none").click();
|
|
}
|
|
if (e.key === 'p' || e.key === 'P')
|
|
{
|
|
$('.add-new-product-dialog-button').not(".d-none").click();
|
|
}
|
|
if (e.key === 'a' || e.key === 'A')
|
|
{
|
|
$('.add-new-product-with-barcode-dialog-button').not(".d-none").click();
|
|
}
|
|
if (e.key === 'c' || e.key === 'C')
|
|
{
|
|
$('.retry-camera-scanning-button').not(".d-none").click();
|
|
}
|
|
if (e.key === 'e' || e.key === 'E')
|
|
{
|
|
$('.add-new-product-plugin-dialog-button').not(".d-none").click();
|
|
}
|
|
});
|
|
}
|
|
else
|
|
{
|
|
Grocy.Components.ProductAmountPicker.Reset();
|
|
Grocy.Components.ProductPicker.Clear();
|
|
Grocy.FrontendHelpers.ValidateForm('consume-form');
|
|
Grocy.Components.ProductPicker.ShowCustomError(__t('This product is not in stock'));
|
|
Grocy.Components.ProductPicker.GetInputElement().focus();
|
|
}
|
|
},
|
|
function(xhr)
|
|
{
|
|
console.error(xhr);
|
|
}
|
|
);
|
|
},
|
|
function(xhr)
|
|
{
|
|
console.error(xhr);
|
|
}
|
|
);
|
|
}
|
|
}
|
|
});
|
|
|
|
$(document).on("Grocy.BarcodeScanned", function(e, barcode, target)
|
|
{
|
|
if (!(target == "@productpicker" || target == "undefined" || target == undefined)) // Default target
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Don't know why the blur event does not fire immediately ... this works...
|
|
Grocy.Components.ProductPicker.GetInputElement().focusout();
|
|
Grocy.Components.ProductPicker.GetInputElement().focus();
|
|
Grocy.Components.ProductPicker.GetInputElement().blur();
|
|
|
|
Grocy.Components.ProductPicker.GetInputElement().val(barcode);
|
|
|
|
setTimeout(function()
|
|
{
|
|
Grocy.Components.ProductPicker.GetInputElement().focusout();
|
|
Grocy.Components.ProductPicker.GetInputElement().focus();
|
|
Grocy.Components.ProductPicker.GetInputElement().blur();
|
|
}, 500);
|
|
});
|
|
|
|
$(document).on("shown.bs.modal", function(e)
|
|
{
|
|
$(".modal-footer").addClass("d-block").addClass("d-sm-flex");
|
|
$(".modal-footer").find("button").addClass("mt-2").addClass("mt-sm-0");
|
|
})
|
|
|
|
// Make that ENTER behaves the same like TAB (trigger blur to start workflows, but only when the dropdown is not opened)
|
|
$('#product_id_text_input').keydown(function(event)
|
|
{
|
|
if (event.keyCode === 13) // Enter
|
|
{
|
|
if (Grocy.Components.ProductPicker.GetPicker().hasClass("combobox-menu-visible"))
|
|
{
|
|
return;
|
|
}
|
|
|
|
$("#product_id_text_input").trigger("blur");
|
|
}
|
|
});
|