forked from Wavyzz/dolibarr
552 lines
22 KiB
PHP
552 lines
22 KiB
PHP
<?php
|
|
/* Copyright (C) 2007-2010 Laurent Destailleur <eldy@users.sourceforge.net>
|
|
* Copyright (C) 2007-2014 Regis Houssin <regis.houssin@capnetworks.com>
|
|
* Copyright (C) 2012 Christophe Battarel <christophe.battarel@altairis.fr>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
* or see http://www.gnu.org/
|
|
*/
|
|
|
|
/**
|
|
* \file htdocs/core/lib/ajax.lib.php
|
|
* \brief Page called by Ajax request for produts
|
|
*/
|
|
|
|
|
|
/**
|
|
* Generic function that return javascript to add to a page to transform a common input field into an autocomplete field by calling an Ajax page (ex: /societe/ajaxcompanies.php).
|
|
* The HTML field must be an input text with id=search_$htmlname.
|
|
* This use the jQuery "autocomplete" function.
|
|
*
|
|
* @param string $selected Preselecte value
|
|
* @param string $htmlname HTML name of input field
|
|
* @param string $url Url for request: /path/page.php. Must return a json array ('key'=>id, 'value'=>String shown into input field once selected, 'label'=>String shown into combo list)
|
|
* @param string $urloption More parameters on URL request
|
|
* @param int $minLength Minimum number of chars to trigger that Ajax search
|
|
* @param int $autoselect Automatic selection if just one value
|
|
* @param array $ajaxoptions Multiple options array (Ex: array('update'=>array('field1','field2'...)) will reset field1 and field2 once select done)
|
|
* @return string Script
|
|
*/
|
|
function ajax_autocompleter($selected, $htmlname, $url, $urloption='', $minLength=2, $autoselect=0, $ajaxoptions=array())
|
|
{
|
|
if (empty($minLength)) $minLength=1;
|
|
|
|
$script = '<input type="hidden" name="'.$htmlname.'" id="'.$htmlname.'" value="'.$selected.'" />';
|
|
|
|
$script.= '<script type="text/javascript">';
|
|
$script.= '$(document).ready(function() {
|
|
var autoselect = '.$autoselect.';
|
|
var options = '.json_encode($ajaxoptions).';
|
|
|
|
// Remove product id before select another product
|
|
// use keyup instead change to avoid loosing the product id
|
|
$("input#search_'.$htmlname.'").keydown(function() {
|
|
//console.log(\'purge_id_after_keydown\');
|
|
$("#'.$htmlname.'").val("");
|
|
});
|
|
$("input#search_'.$htmlname.'").change(function() {
|
|
//console.log(\'change\');
|
|
$("#'.$htmlname.'").trigger("change");
|
|
});
|
|
// Check when keyup
|
|
$("input#search_'.$htmlname.'").keyup(function() {
|
|
//console.log(\'keyup\');
|
|
if ($(this).val().length == 0)
|
|
{
|
|
$("#search_'.$htmlname.'").val("");
|
|
$("#'.$htmlname.'").val("").trigger("change");
|
|
if (options.option_disabled) {
|
|
$("#" + options.option_disabled).removeAttr("disabled");
|
|
}
|
|
if (options.disabled) {
|
|
$.each(options.disabled, function(key, value) {
|
|
$("#" + value).removeAttr("disabled");
|
|
});
|
|
}
|
|
if (options.update) {
|
|
$.each(options.update, function(key, value) {
|
|
$("#" + key).val("").trigger("change");
|
|
});
|
|
}
|
|
if (options.show) {
|
|
$.each(options.show, function(key, value) {
|
|
$("#" + value).hide().trigger("hide");
|
|
});
|
|
}
|
|
if (options.update_textarea) {
|
|
$.each(options.update_textarea, function(key, value) {
|
|
if (typeof CKEDITOR == "object" && typeof CKEDITOR.instances != "undefined" && CKEDITOR.instances[key] != "undefined") {
|
|
CKEDITOR.instances[key].setData("");
|
|
} else {
|
|
$("#" + key).html("");
|
|
}
|
|
});
|
|
}
|
|
}
|
|
});
|
|
$("input#search_'.$htmlname.'").autocomplete({
|
|
source: function( request, response ) {
|
|
$.get("'.$url.($urloption?'?'.$urloption:'').'", { '.$htmlname.': request.term }, function(data){
|
|
if (data != null)
|
|
{
|
|
response($.map( data, function(item) {
|
|
if (autoselect == 1 && data.length == 1) {
|
|
$("#search_'.$htmlname.'").val(item.value);
|
|
$("#'.$htmlname.'").val(item.key).trigger("change");
|
|
}
|
|
var label = item.label.toString();
|
|
var update = {};
|
|
if (options.update) {
|
|
$.each(options.update, function(key, value) {
|
|
update[key] = item[value];
|
|
});
|
|
}
|
|
var textarea = {};
|
|
if (options.update_textarea) {
|
|
$.each(options.update_textarea, function(key, value) {
|
|
textarea[key] = item[value];
|
|
});
|
|
}
|
|
return { label: label, value: item.value, id: item.key, update: update, textarea: textarea, disabled: item.disabled }
|
|
}));
|
|
}
|
|
else console.error("Error: Ajax url '.$url.($urloption?'?'.$urloption:'').' has returned an empty page. Should be an empty json array.");
|
|
}, "json");
|
|
},
|
|
dataType: "json",
|
|
minLength: '.$minLength.',
|
|
select: function( event, ui ) { // Function ran when new value is selected into javascript combo
|
|
//console.log(\'set value of id with \'+ui.item.id);
|
|
$("#'.$htmlname.'").val(ui.item.id).trigger("change"); // Select new value
|
|
// Disable an element
|
|
if (options.option_disabled) {
|
|
if (ui.item.disabled) {
|
|
$("#" + options.option_disabled).prop("disabled", true);
|
|
if (options.error) {
|
|
$.jnotify(options.error, "error", true); // Output with jnotify the error message
|
|
}
|
|
if (options.warning) {
|
|
$.jnotify(options.warning, "warning", false); // Output with jnotify the warning message
|
|
}
|
|
} else {
|
|
$("#" + options.option_disabled).removeAttr("disabled");
|
|
}
|
|
}
|
|
if (options.disabled) {
|
|
$.each(options.disabled, function(key, value) {
|
|
$("#" + value).prop("disabled", true);
|
|
});
|
|
}
|
|
if (options.show) {
|
|
$.each(options.show, function(key, value) {
|
|
$("#" + value).show().trigger("show");
|
|
});
|
|
}
|
|
// Update an input
|
|
if (ui.item.update) {
|
|
// loop on each "update" fields
|
|
$.each(ui.item.update, function(key, value) {
|
|
$("#" + key).val(value).trigger("change");
|
|
});
|
|
}
|
|
if (ui.item.textarea) {
|
|
$.each(ui.item.textarea, function(key, value) {
|
|
if (typeof CKEDITOR == "object" && typeof CKEDITOR.instances != "undefined" && CKEDITOR.instances[key] != "undefined") {
|
|
CKEDITOR.instances[key].setData(value);
|
|
CKEDITOR.instances[key].focus();
|
|
} else {
|
|
$("#" + key).html(value);
|
|
$("#" + key).focus();
|
|
}
|
|
});
|
|
}
|
|
$("#search_'.$htmlname.'").trigger("change"); // To tell that input text field was modified
|
|
}
|
|
,delay: 500
|
|
}).data( "autocomplete" )._renderItem = function( ul, item ) {
|
|
return $( "<li></li>" )
|
|
.data( "item.autocomplete", item )
|
|
.append( \'<a><span class="tag">\' + item.label + "</span></a>" )
|
|
.appendTo(ul);
|
|
};
|
|
|
|
});';
|
|
$script.= '</script>';
|
|
|
|
return $script;
|
|
}
|
|
|
|
/**
|
|
* Generic function that return javascript to add to a page to transform a common input field into an autocomplete field by calling an Ajax page (ex: core/ajax/ziptown.php).
|
|
* The Ajax page can also returns several values (json format) to fill several input fields.
|
|
* The HTML field must be an input text with id=$htmlname.
|
|
* This use the jQuery "autocomplete" function.
|
|
*
|
|
* @param string $htmlname HTML name of input field
|
|
* @param string $fields Other fields to autocomplete
|
|
* @param string $url URL for ajax request : /chemin/fichier.php
|
|
* @param string $option More parameters on URL request
|
|
* @param int $minLength Minimum number of chars to trigger that Ajax search
|
|
* @param int $autoselect Automatic selection if just one value
|
|
* @return string Script
|
|
*/
|
|
function ajax_multiautocompleter($htmlname, $fields, $url, $option='', $minLength=2, $autoselect=0)
|
|
{
|
|
$script = '<!-- Autocomplete -->'."\n";
|
|
$script.= '<script type="text/javascript">';
|
|
$script.= 'jQuery(document).ready(function() {
|
|
var fields = '.json_encode($fields).';
|
|
var nboffields = fields.length;
|
|
var autoselect = '.$autoselect.';
|
|
//alert(fields + " " + nboffields);
|
|
|
|
jQuery("input#'.$htmlname.'").autocomplete({
|
|
dataType: "json",
|
|
minLength: '.$minLength.',
|
|
source: function( request, response ) {
|
|
jQuery.getJSON( "'.$url.($option?'?'.$option:'').'", { '.$htmlname.': request.term }, function(data){
|
|
response( jQuery.map( data, function( item ) {
|
|
if (autoselect == 1 && data.length == 1) {
|
|
jQuery("#'.$htmlname.'").val(item.value);
|
|
// TODO move this to specific request
|
|
if (item.states) {
|
|
jQuery("#state_id").html(item.states);
|
|
}
|
|
for (i=0;i<nboffields;i++) {
|
|
if (item[fields[i]]) { // If defined
|
|
//alert(item[fields[i]]);
|
|
jQuery("#" + fields[i]).val(item[fields[i]]);
|
|
}
|
|
}
|
|
}
|
|
return item
|
|
}));
|
|
});
|
|
},
|
|
select: function( event, ui ) {
|
|
for (i=0;i<nboffields;i++) {
|
|
//alert(fields[i] + " = " + ui.item[fields[i]]);
|
|
if (fields[i]=="selectcountry_id")
|
|
{
|
|
if (ui.item[fields[i]] > 0) // Do not erase country if unknown
|
|
{
|
|
jQuery("#" + fields[i]).val(ui.item[fields[i]]);
|
|
// If we set new country and new state, we need to set a new list of state to allow change
|
|
if (ui.item.states && ui.item["state_id"] != jQuery("#state_id").value) {
|
|
jQuery("#state_id").html(ui.item.states);
|
|
}
|
|
}
|
|
}
|
|
else if (fields[i]=="state_id" || fields[i]=="state_id")
|
|
{
|
|
if (ui.item[fields[i]] > 0) // Do not erase state if unknown
|
|
{
|
|
jQuery("#" + fields[i]).val(ui.item[fields[i]]); // This may fails if not correct country
|
|
}
|
|
}
|
|
else if (ui.item[fields[i]]) { // If defined
|
|
//alert(fields[i]);
|
|
//alert(ui.item[fields[i]]);
|
|
jQuery("#" + fields[i]).val(ui.item[fields[i]]);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
});';
|
|
$script.= '</script>';
|
|
|
|
return $script;
|
|
}
|
|
|
|
/**
|
|
* Show an ajax dialog
|
|
*
|
|
* @param string $title Title of dialog box
|
|
* @param string $message Message of dialog box
|
|
* @param int $w Width of dialog box
|
|
* @param int $h height of dialog box
|
|
* @return void
|
|
*/
|
|
function ajax_dialog($title,$message,$w=350,$h=150)
|
|
{
|
|
global $langs;
|
|
|
|
$msg= '<div id="dialog-info" title="'.dol_escape_htmltag($title).'">';
|
|
$msg.= $message;
|
|
$msg.= '</div>'."\n";
|
|
$msg.= '<script type="text/javascript">
|
|
jQuery(function() {
|
|
jQuery("#dialog-info").dialog({
|
|
resizable: false,
|
|
height:'.$h.',
|
|
width:'.$w.',
|
|
modal: true,
|
|
buttons: {
|
|
Ok: function() {
|
|
jQuery(this).dialog(\'close\');
|
|
}
|
|
}
|
|
});
|
|
});
|
|
</script>';
|
|
|
|
$msg.= "\n";
|
|
|
|
return $msg;
|
|
}
|
|
|
|
/**
|
|
* Convert a html select field into an ajax combobox.
|
|
* Use ajax_combobox() only for small combo list! If not, use instead ajax_autocompleter().
|
|
* TODO: It is used when COMPANY_USE_SEARCH_TO_SELECT and CONTACT_USE_SEARCH_TO_SELECT are set by html.formcompany.class.php. Should use ajax_autocompleter instead like done by html.form.class.php for select_produits.
|
|
*
|
|
* @param string $htmlname Name of html select field
|
|
* @param array $events More events option. Example: array(array('method'=>'getContacts', 'url'=>dol_buildpath('/core/ajax/contacts.php',1), 'htmlname'=>'contactid', 'params'=>array('add-customer-contact'=>'disabled')))
|
|
* @param int $minLengthToAutocomplete Minimum length of input string to start autocomplete
|
|
* @param int $forcefocus Force focus on field
|
|
* @return string Return html string to convert a select field into a combo, or '' if feature has been disabled for some reason.
|
|
*/
|
|
function ajax_combobox($htmlname, $events=array(), $minLengthToAutocomplete=0, $forcefocus=0)
|
|
{
|
|
global $conf;
|
|
|
|
//if (! empty($conf->browser->phone)) return ''; // combobox disabled for smartphones (does not works)
|
|
if (! empty($conf->dol_use_jmobile)) return ''; // select2 works with jmobile but it breaks the autosize feature of jmobile.
|
|
if (! empty($conf->global->MAIN_DISABLE_AJAX_COMBOX)) return '';
|
|
if (empty($conf->use_javascript_ajax)) return '';
|
|
|
|
if (empty($minLengthToAutocomplete)) $minLengthToAutocomplete=0;
|
|
|
|
$tmpplugin='select2';
|
|
$msg='<!-- JS CODE TO ENABLE '.$tmpplugin.' for id '.$htmlname.' -->
|
|
<script type="text/javascript">
|
|
$(document).ready(function () {
|
|
$(\''.(preg_match('/^\./',$htmlname)?$htmlname:'#'.$htmlname).'\').'.$tmpplugin.'({
|
|
dir: \'ltr\',
|
|
width: \'resolve\', /* off or resolve */
|
|
minimumInputLength: '.$minLengthToAutocomplete.'
|
|
})';
|
|
if ($forcefocus) $msg.= '.select2(\'focus\')';
|
|
$msg.= ';'."\n";
|
|
|
|
if (count($event))
|
|
{
|
|
$msg.= '
|
|
jQuery("#'.$htmlname.'").change(function () {
|
|
var obj = '.json_encode($events).';
|
|
$.each(obj, function(key,values) {
|
|
if (values.method.length) {
|
|
runJsCodeForEvent'.$htmlname.'(values);
|
|
}
|
|
});
|
|
});
|
|
|
|
function runJsCodeForEvent'.$htmlname.'(obj) {
|
|
var id = $("#'.$htmlname.'").val();
|
|
var method = obj.method;
|
|
var url = obj.url;
|
|
var htmlname = obj.htmlname;
|
|
var showempty = obj.showempty;
|
|
$.getJSON(url,
|
|
{
|
|
action: method,
|
|
id: id,
|
|
htmlname: htmlname,
|
|
showempty: showempty
|
|
},
|
|
function(response) {
|
|
$.each(obj.params, function(key,action) {
|
|
if (key.length) {
|
|
var num = response.num;
|
|
if (num > 0) {
|
|
$("#" + key).removeAttr(action);
|
|
} else {
|
|
$("#" + key).attr(action, action);
|
|
}
|
|
}
|
|
});
|
|
$("select#" + htmlname).html(response.value);
|
|
if (response.num) {
|
|
var selecthtml_str = response.value;
|
|
var selecthtml_dom=$.parseHTML(selecthtml_str);
|
|
$("#inputautocomplete"+htmlname).val(selecthtml_dom[0][0].innerHTML);
|
|
} else {
|
|
$("#inputautocomplete"+htmlname).val("");
|
|
}
|
|
$("select#" + htmlname).change(); /* Trigger event change */
|
|
}
|
|
);
|
|
}';
|
|
}
|
|
|
|
$msg.= '});'."\n";
|
|
$msg.= "</script>\n";
|
|
|
|
return $msg;
|
|
}
|
|
|
|
/**
|
|
* On/off button for constant
|
|
*
|
|
* @param string $code Name of constant
|
|
* @param array $input Array of type->list of CSS element to switch. Example: array('disabled'=>array(0=>'cssid'))
|
|
* @param int $entity Entity to set
|
|
* @param int $revertonoff Revert on/off
|
|
* @param bool $strict Use only "disabled" with delConstant and "enabled" with setConstant
|
|
* @return string
|
|
*/
|
|
function ajax_constantonoff($code, $input=array(), $entity=null, $revertonoff=0, $strict=0)
|
|
{
|
|
global $conf, $langs;
|
|
|
|
$entity = ((isset($entity) && is_numeric($entity) && $entity >= 0) ? $entity : $conf->entity);
|
|
|
|
if (empty($conf->use_javascript_ajax))
|
|
{
|
|
if (empty($conf->global->$code)) print '<a href="'.$_SERVER['PHP_SELF'].'?action=set_'.$code.'&entity='.$entity.'">'.img_picto($langs->trans("Disabled"),'off').'</a>';
|
|
else print '<a href="'.$_SERVER['PHP_SELF'].'?action=del_'.$code.'&entity='.$entity.'">'.img_picto($langs->trans("Enabled"),'on').'</a>';
|
|
}
|
|
else
|
|
{
|
|
$out= "\n<!-- Ajax code to switch constant ".$code." -->".'
|
|
<script type="text/javascript">
|
|
$(document).ready(function() {
|
|
var input = '.json_encode($input).';
|
|
var url = \''.DOL_URL_ROOT.'/core/ajax/constantonoff.php\';
|
|
var code = \''.$code.'\';
|
|
var entity = \''.$entity.'\';
|
|
var strict = \''.$strict.'\';
|
|
var yesButton = "'.dol_escape_js($langs->transnoentities("Yes")).'";
|
|
var noButton = "'.dol_escape_js($langs->transnoentities("No")).'";
|
|
|
|
// Set constant
|
|
$("#set_" + code).click(function() {
|
|
if (input.alert && input.alert.set) {
|
|
if (input.alert.set.yesButton) yesButton = input.alert.set.yesButton;
|
|
if (input.alert.set.noButton) noButton = input.alert.set.noButton;
|
|
confirmConstantAction("set", url, code, input, input.alert.set, entity, yesButton, noButton, strict);
|
|
} else {
|
|
setConstant(url, code, input, entity);
|
|
}
|
|
});
|
|
|
|
// Del constant
|
|
$("#del_" + code).click(function() {
|
|
if (input.alert && input.alert.del) {
|
|
if (input.alert.del.yesButton) yesButton = input.alert.del.yesButton;
|
|
if (input.alert.del.noButton) noButton = input.alert.del.noButton;
|
|
confirmConstantAction("del", url, code, input, input.alert.del, entity, yesButton, noButton, strict);
|
|
} else {
|
|
delConstant(url, code, input, entity);
|
|
}
|
|
});
|
|
});
|
|
</script>'."\n";
|
|
|
|
$out.= '<div id="confirm_'.$code.'" title="" style="display: none;"></div>';
|
|
$out.= '<span id="set_'.$code.'" class="linkobject '.(! empty($conf->global->$code)?'hideobject':'').'">'.($revertonoff?img_picto($langs->trans("Enabled"),'switch_on'):img_picto($langs->trans("Disabled"),'switch_off')).'</span>';
|
|
$out.= '<span id="del_'.$code.'" class="linkobject '.(! empty($conf->global->$code)?'':'hideobject').'">'.($revertonoff?img_picto($langs->trans("Disabled"),'switch_off'):img_picto($langs->trans("Enabled"),'switch_on')).'</span>';
|
|
$out.="\n";
|
|
}
|
|
|
|
return $out;
|
|
}
|
|
|
|
/**
|
|
* On/off button for object
|
|
*
|
|
* @param int $object Id product to set
|
|
* @param string $code Name of constant : status or status_buy for product by example
|
|
* @param string $field Name of database field : tosell or tobuy for product by example
|
|
* @param string $text_on Text if on
|
|
* @param string $text_off Text if off
|
|
* @param array $input Array of type->list of CSS element to switch. Example: array('disabled'=>array(0=>'cssid'))
|
|
* @return void
|
|
*/
|
|
function ajax_object_onoff($object, $code, $field, $text_on, $text_off, $input=array())
|
|
{
|
|
global $langs;
|
|
|
|
$out= '<script type="text/javascript">
|
|
$(function() {
|
|
var input = '.json_encode($input).';
|
|
|
|
// Set constant
|
|
$("#set_'.$code.'_'.$object->id.'").click(function() {
|
|
$.get( "'.DOL_URL_ROOT.'/core/ajax/objectonoff.php", {
|
|
action: \'set\',
|
|
field: \''.$field.'\',
|
|
value: \'1\',
|
|
element: \''.$object->element.'\',
|
|
id: \''.$object->id.'\'
|
|
},
|
|
function() {
|
|
$("#set_'.$code.'_'.$object->id.'").hide();
|
|
$("#del_'.$code.'_'.$object->id.'").show();
|
|
// Enable another element
|
|
if (input.disabled && input.disabled.length > 0) {
|
|
$.each(input.disabled, function(key,value) {
|
|
$("#" + value).removeAttr("disabled");
|
|
if ($("#" + value).hasClass("butActionRefused") == true) {
|
|
$("#" + value).removeClass("butActionRefused");
|
|
$("#" + value).addClass("butAction");
|
|
}
|
|
});
|
|
// Show another element
|
|
} else if (input.showhide && input.showhide.length > 0) {
|
|
$.each(input.showhide, function(key,value) {
|
|
$("#" + value).show();
|
|
});
|
|
}
|
|
});
|
|
});
|
|
|
|
// Del constant
|
|
$("#del_'.$code.'_'.$object->id.'").click(function() {
|
|
$.get( "'.DOL_URL_ROOT.'/core/ajax/objectonoff.php", {
|
|
action: \'set\',
|
|
field: \''.$field.'\',
|
|
value: \'0\',
|
|
element: \''.$object->element.'\',
|
|
id: \''.$object->id.'\'
|
|
},
|
|
function() {
|
|
$("#del_'.$code.'_'.$object->id.'").hide();
|
|
$("#set_'.$code.'_'.$object->id.'").show();
|
|
// Disable another element
|
|
if (input.disabled && input.disabled.length > 0) {
|
|
$.each(input.disabled, function(key,value) {
|
|
$("#" + value).prop("disabled", true);
|
|
if ($("#" + value).hasClass("butAction") == true) {
|
|
$("#" + value).removeClass("butAction");
|
|
$("#" + value).addClass("butActionRefused");
|
|
}
|
|
});
|
|
// Hide another element
|
|
} else if (input.showhide && input.showhide.length > 0) {
|
|
$.each(input.showhide, function(key,value) {
|
|
$("#" + value).hide();
|
|
});
|
|
}
|
|
});
|
|
});
|
|
});
|
|
</script>';
|
|
$out.= '<span id="set_'.$code.'_'.$object->id.'" class="linkobject '.($object->$code==1?'hideobject':'').'">'.img_picto($langs->trans($text_off),'switch_off').'</span>';
|
|
$out.= '<span id="del_'.$code.'_'.$object->id.'" class="linkobject '.($object->$code==1?'':'hideobject').'">'.img_picto($langs->trans($text_on),'switch_on').'</span>';
|
|
|
|
return $out;
|
|
}
|
|
|