21include_once DOL_DOCUMENT_ROOT.
'/core/lib/admin.lib.php';
55 public $file_source_url;
82 public $dolistore_api_url;
86 public $dolistore_api_key;
91 public $dolistoreApiStatus;
96 public $dolistoreApiError;
101 public $githubFileStatus;
106 public $githubFileError;
116 public $numberOfProviders;
126 public $numberTotalOfProducts;
131 public $numberTotalOfPages;
136 public $numberOfProducts;
147 $this->debug_api = $debug;
149 $this->url = DOL_URL_ROOT.
'/admin/modules.php?mode=marketplace';
152 $this->dolistore_api_url =
getDolGlobalString(
'MAIN_MODULE_DOLISTORE_API_SRV',
'https://www.dolistore.com/api/');
153 $this->dolistore_api_key =
getDolGlobalString(
'MAIN_MODULE_DOLISTORE_API_KEY',
'dolistorepublicapi');
154 $this->shop_url =
getDolGlobalString(
'MAIN_MODULE_DOLISTORE_SHOP_URL',
'https://www.dolistore.com');
157 $this->file_source_url =
"https://raw.githubusercontent.com/Dolibarr/dolibarr-community-modules/refs/heads/main/index.yaml";
158 $this->cache_file = DOL_DATA_ROOT.
'/admin/temp/remote_github_modules_file.yaml';
160 $lang = $langs->defaultlang;
161 $lang_array = array(
'en_US',
'fr_FR',
'es_ES',
'it_IT',
'de_DE');
162 if (!in_array($lang, $lang_array)) {
178 $cachedelayforgithubrepo =
getDolGlobalInt(
'MAIN_REMOTE_GITHUBREPO_CACHE_DELAY', 86400);
182 $this->githubFileError = $this->error;
183 $this->githubFileStatus =
dol_is_file($this->cache_file) ? 1 : 0;
192 $this->numberOfProviders = $this->dolistoreApiStatus + $this->githubFileStatus;
202 public function callApi($resource, $options =
false)
205 if (empty($this->dolistore_api_key) || empty($this->dolistore_api_url)) {
206 return array(
'status_code' => 0,
'response' =>
null);
213 $httpheader = array(
'DOLAPIKEY: '.$this->dolistore_api_key);
214 if ($basicAuthLogin) {
215 $httpheader[] =
'Authorization: Basic '.base64_encode($basicAuthLogin.
':'.$basicAuthPassword);
218 $url = $this->dolistore_api_url . (preg_match(
'/\/$/', $this->dolistore_api_url) ?
'' :
'/') . $resource;
220 $options[
'apikey'] = $this->dolistore_api_key;
223 $url .=
'?' . http_build_query($options);
226 $response =
getURLContent($url,
'GET',
'', 1, $httpheader, array(
'https'), 0, -1, 5, 5);
228 $status_code = $response[
'http_code'];
231 if ($status_code == 200) {
232 $body = $response[
'content'];
233 $body = json_decode($body,
true);
234 $returnarray = array(
235 'status_code' => $status_code,
239 $returnarray = array(
240 'status_code' => $status_code,
243 if (!empty($response[
'curl_error_no'])) {
244 $returnarray[
'curl_error_no'] = $response[
'curl_error_no'];
246 if (!empty($response[
'curl_error_msg'])) {
247 $returnarray[
'curl_error_msg'] = $response[
'curl_error_msg'];
264 if (!empty($this->cache_file) && file_exists($this->cache_file)) {
265 dol_syslog(__METHOD__ .
" - Loading cache file: " . $this->cache_file, LOG_DEBUG);
267 $content = file_get_contents($this->cache_file);
268 if ($content !==
false) {
269 $modules = $this->
readYaml($content);
271 dol_syslog(__METHOD__ .
" - Error reading cache file", LOG_ERR);
286 $organized_tree = array();
290 'lang' => $this->lang
295 $resCategories = $this->
callApi(
'categories', $data);
296 if (isset($resCategories[
'response']) && is_array($resCategories[
'response'])) {
297 $organized_tree = $resCategories[
'response'];
303 foreach ($organized_tree as $key => $value) {
304 if ($value[
'label'] !=
"Versions" && $value[
'label'] !=
"Specials") {
305 $html .=
'<li' . ($current == $value[
'rowid'] ?
' class="active"' :
'') .
'>';
306 $html .=
'<a href="?mode=marketplace&categorie=' . $value[
'rowid'] .
'">' . $value[
'label'] .
'</a>';
307 if (isset($value[
'children'])) {
309 usort($value[
'children'], $this->
buildSorter(
'position'));
310 foreach ($value[
'children'] as $key_children => $value_children) {
311 $html .=
'<li' . ($current == $value_children[
'rowid'] ?
' class="active"' :
'') .
'>';
312 $html .=
'<a href="?mode=marketplace&categorie=' . $value_children[
'rowid'] .
'" title="' .
dol_escape_htmltag(strip_tags($value_children[
'description'])) .
'">' . $value_children[
'label'] .
'</a>';
333 $langs->load(
"products");
336 $last_month =
dol_now() - (30 * 24 * 60 * 60);
337 $dolibarrversiontouse = DOL_VERSION;
339 $this->products = array();
341 $this->categorie = $options[
'categorie'] ?? 0;
342 $this->per_page = $options[
'per_page'] ?? 11;
343 $this->no_page = $options[
'no_page'] ?? 1;
344 $this->search = $options[
'search'] ??
'';
346 $this->per_page = 11;
349 if (!empty($this->search) && strlen(str_replace(
' ',
'', (
string) $this->search)) < 2) {
350 $html .=
'<tr class=""><td colspan="3" class="center">';
352 $html .= $langs->trans(
"SearchStringMinLength").
'...';
354 $html .=
'</td></tr>';
359 'categorieid' => $this->categorie,
360 'limit' => $this->per_page,
361 'page' => $this->no_page,
362 'search' => $this->search,
363 'lang' => $this->lang
367 $this->numberTotalOfProducts = 0;
370 if ($this->categorie == 87) {
371 $html =
'<div class="shop-container">
372 <div class="shop-image">
373 <a href="https://merch.dolibarr.org/" target="_blank">
374 <img src="https://www.dolistore.com/medias/image/marketplace/img/goodies-shop.jpg" width="50%" alt="DoliStore Merch and Gifts" />
375 <div class="shop-overlay">
376 <button target="new" class="shop-button">'.$langs->trans(
"GoodiesButtonTitle").
' <i class="icon-chevron-right"></i></button>
387 $dolistoreProducts = array();
388 $dolistoreProductsTotal = 0;
389 if ($this->dolistoreApiStatus > 0 &&
getDolGlobalInt(
'MAIN_ENABLE_EXTERNALMODULES_DOLISTORE')) {
390 $getDolistoreProducts = $this->
callApi(
'products', $data);
392 if (!isset($getDolistoreProducts[
'response']) || !is_array($getDolistoreProducts[
'response']) || ($getDolistoreProducts[
'status_code'] != 200 && $getDolistoreProducts[
'status_code'] != 201)) {
393 $dolistoreProducts = array();
394 $dolistoreProductsTotal = 0;
396 $dolistoreProducts = $this->
adaptData($getDolistoreProducts[
'response'][
'products'],
'dolistore');
397 $dolistoreProductsTotal = (int) $getDolistoreProducts[
'response'][
'total'];
398 $this->numberTotalOfProducts += $dolistoreProductsTotal;
404 $fileProducts = array();
405 $fileProductsTotal = 0;
406 if (!empty($this->githubFileStatus) &&
getDolGlobalInt(
'MAIN_ENABLE_EXTERNALMODULES_COMMUNITY')) {
409 $fileProducts = $this->
adaptData($fileProducts,
'githubcommunity');
411 $fileProducts = $this->
applyFilters($fileProducts, $data);
413 $fileProductsTotal = $fileProducts[
'total'];
415 $this->numberTotalOfProducts += $fileProductsTotal;
417 $fileProducts = $fileProducts[
'data'];
421 $this->numberTotalOfPages = (int) ceil(max($fileProductsTotal / $this->per_page, $dolistoreProductsTotal / $this->per_page));
424 $this->products = $dolistoreProducts;
425 foreach ($fileProducts as $fileProduct) {
426 $id = $fileProduct[
'id'];
428 if (empty($this->products[
$id])) {
429 array_unshift($this->products, $fileProduct);
431 $this->products[
$id] = $fileProduct;
432 $this->products[
$id][
'category'] = $fileProduct[
'category'];
435 array_unshift($this->products, $fileProduct);
441 foreach ($this->products as $product) {
446 if ($last_month < strtotime($product[
'datec']) && $product[
"status"] !=
'soon' && $product[
"status"] !=
'development' && $product[
"status"] !=
'experimental') {
447 $newapp .=
'<span class="newApp" title="'.$product[
'tms'].
'">'.$langs->trans(
'New').
'</span> ';
451 if ($newapp ==
'' && $last_month < strtotime($product[
'tms']) && $product[
"status"] !=
'soon' && $product[
"status"] !=
'development' && $product[
"status"] !=
'experimental') {
452 $newapp .=
'<span class="updatedApp" title="'.$product[
'tms'].
'">'.$langs->trans(
'UpdatedRecently').
'</span> ';
456 if ($product[
"cover_photo_url"] !=
'' && $product[
"cover_photo_url"] !=
'#') {
457 $images =
'<a href="'.$product[
"cover_photo_url"].
'" class="documentpreview" target="_blank" rel="noopener noreferrer" mime="image/png" title="'.
dol_escape_htmltag($product[
"label"].
', '.$langs->trans(
'Version').
' '.$product[
"module_version"]).
'">';
458 $images .=
'<img class="imgstore" src="'.$product[
"cover_photo_url"].
'" alt="" /></a>';
460 $images =
'<img class="imgstore" src="'.DOL_URL_ROOT.
'/public/theme/common/nophoto.png" />';
466 if ($product[
"status"] ==
'soon' || $product[
"status"] ==
'development' || $product[
"status"] ==
'experimental') {
467 $version =
'<span class="warning">'.$langs->trans(
"NotYetAvailable").
' - '.$langs->trans(
"StillInDevelopment").
'</span>';
468 $compatible =
'NotCompatible';
469 } elseif ($this->
versionCompare($product[
"dolibarr_min"], $dolibarrversiontouse) <= 0) {
470 if (!empty($product[
"dolibarr_max"]) && $product[
"dolibarr_max"] !=
'auto' && $product[
"dolibarr_max"] !=
'unknown' && $this->
versionCompare($product[
"dolibarr_max"], $dolibarrversiontouse) >= 0) {
472 $version =
'<span class="compatible hideonsmartphone">'.$langs->trans(
474 $dolibarrversiontouse,
475 $product[
"dolibarr_min"],
476 $product[
"dolibarr_max"]
481 $version =
'<span class="warning">'.$langs->trans(
483 $dolibarrversiontouse,
484 $product[
"dolibarr_min"],
485 $product[
"dolibarr_max"]
487 $compatible =
'NotCompatible';
490 if ($product[
"dolibarr_min"] ==
'auto' || $product[
"dolibarr_min"] !=
'unknown') {
492 $version =
'<span class="warning">'.$langs->trans(
494 $dolibarrversiontouse,
495 $product[
"dolibarr_min"],
496 $product[
"dolibarr_max"]
498 $compatible =
'NotCompatible';
501 $version =
'<span class="compatibleafterupdate">'.$langs->trans(
502 'CompatibleAfterUpdate',
503 $dolibarrversiontouse,
504 $product[
"dolibarr_min"],
505 $product[
"dolibarr_max"]
507 $compatible =
'NotCompatible';
513 if (array_key_exists(
'price_ht', $product) &&
price2num($product[
"price_ht"]) > 0) {
514 $price =
'<h3>'.price(
price2num($product[
"price_ht"],
'MT'), 0, $langs, 1, -1, -1,
'EUR').
' '.$langs->trans(
"HT").
'</h3>';
516 $download_link =
'<a class="paddingleft paddingright valignmiddle" target="_blank" title="'.$langs->trans(
"View").
'" href="'.$this->shop_url.
'/product.php?id='.((int) $product[
'id']).
'">';
517 $download_link .=
img_picto(
'',
'url',
'class="size2x paddingright"');
518 $download_link .=
'</a>';
520 $download_link =
'#';
521 if ($product[
'source'] ===
'dolistore') {
522 $urlview = $this->shop_url.
'/product.php?id='.((int) $product[
"id"]);
523 $price =
'<h3><a href="'.$urlview.
'" target="_blank">'.$langs->trans(
'SeeOnDoliStore').
'</a></h3>';
524 } elseif ($product[
'source'] ===
'githubcommunity') {
525 if (array_key_exists(
'price_ht', $product) && empty($product[
'price_ht'])) {
526 if ($product[
'status'] ==
'soon') {
527 $price =
'<h3>'.$langs->trans(
'StillInDevelopment').
'</h3>';
529 $price =
'<h3>'.$langs->trans(
'Free').
'</h3>';
532 if ($product[
"dolistore-download"]) {
533 $price =
'<h3><a href="'.$product[
"dolistore-download"].
'" target="_blank">'.$langs->trans(
'SeeOnDoliStore').
'</a></h3>';
535 $price =
'<h3>'.$langs->trans(
'Unknown').
'</h3>';
539 $price =
'<h3>'.$langs->trans(
'Unknown').
'</h3>';
542 if ($product[
'source'] ===
'githubcommunity') {
543 $download_link =
'<a class="paddingleft paddingright valignmiddle" target="_blank" title="'.$langs->trans(
"Sources").
'" href="'.$product[
"link"].
'">';
544 $download_link .=
img_picto(
'',
'file-code',
'class="size2x paddingright colorgrey"');
545 $download_link .=
'</a>';
547 $urlview = $product[
"dolistore-download"];
549 $download_link .=
'<a class="paddingleft paddingright valignmiddle" target="_blank" title="'.$langs->trans(
"View").
'" href="'.$urlview.
'" rel="noopener noreferrer">';
550 $download_link .=
img_picto(
'',
'url',
'class="size2x"');
551 $download_link .=
'</a>';
554 if (!empty($product[
'direct-download']) && $product[
'direct-download'] ==
'yes') {
556 if (preg_match(
'/https:.*\?id=(\d+)$/', $urlview, $reg)) {
557 $urldownload =
'https://www.dolistore.com/_service_download.php?t=free&p='.$reg[1];
558 $download_link .=
'<a class="paddingleft paddingright valignmiddle" target="_blank" title="'.$langs->trans(
"Download").
'" href="'.$urldownload.
'" rel="noopener noreferrer">';
559 $download_link .=
img_picto(
'',
'download',
'class="size2x paddingright"');
561 $download_link .=
'</a>';
564 } elseif ($product[
'source'] ===
'dolistore') {
565 $urlview = $this->shop_url.
'/product.php?id='.((int) $product[
"id"]);
566 $urldownload =
'https://www.dolistore.com/_service_download.php?t=free&p=' . $product[
'id'];
567 $download_link =
'<a class="paddingleft paddingright valignmiddle" target="_blank" title="'.$langs->trans(
"View").
'" href="'.$urlview.
'">';
568 $download_link .=
img_picto(
'',
'url',
'class="size2x"');
569 $download_link .=
'</a>';
570 $download_link .=
'<a class="paddingleft paddingright" target="_blank" title="'.$langs->trans(
"Download").
'" href="'.$urldownload.
'" rel="noopener noreferrer">';
571 $download_link .=
img_picto(
'',
'download',
'class="size2x paddingright"');
573 $download_link .=
'</a>';
577 if (($product[
'direct-download'] && $product[
'direct-download'] ==
'yes') || $product[
'source'] ===
'dolistore') {
578 $disableInstall = ($compatible ===
'NotCompatible');
581 $fields = [
'action' =>
'install',
'token' => newToken()];
582 foreach ($product as $key => $value) {
583 $fields[
'producttoinstall['.$key.
']'] = $value;
586 $install_link =
'<button class="valignmiddle ' . ($disableInstall ?
'butActionRefused' :
'butAction') .
' paddingleft paddingright"'
588 . (!$disableInstall ?
' data-confirm' :
'')
589 . (!$disableInstall ?
' data-fields="' .
dol_escape_htmltag(json_encode($fields)) .
'"' :
'')
591 . (!$disableInstall ?
' data-confirm-title="' .
dol_escape_htmltag($langs->trans(
"extModuleConfirmInstallTitle")) .
'"' :
'')
592 . (!$disableInstall ?
' data-confirm-text="' .
dol_escape_htmltag($langs->transnoentities(
593 "extModuleConfirmInstallText",
594 $product[
'label'] ??
'',
595 $product[
'module_version'] ??
'',
596 $product[
'ref'] ??
'',
597 !empty($product[
'tms']) ?
dol_print_date($product[
'tms'],
'%d/%m/%Y') :
''
599 .
'>' . $langs->trans(
"install") .
'</button>';
604 $html .=
'<tr class="'.(getDolOptimizeSmallScreen() ?
'app' :
'app app2').
' oddeven nohover '.
dol_escape_htmltag($compatible).
'">';
607 $html .=
'<td class="center width150"><div class="newAppParent">';
608 $html .= $newapp.$images;
609 $html .=
'</div></td>';
612 $html .=
'<td class="margeCote minwidth400imp"><h2 class="appTitle">';
614 if (!empty($product[
'author']) && $product[
'author'] !=
'unkownauthor') {
615 $html .=
'<span class="small"> - '.img_picto(
'',
'company',
'class="pictofixedwidth"');
616 if (!empty($product[
'author_url'])) {
617 $html .=
'<a href="'.$product[
'author_url'].
'" target="_blank">'.$product[
'author'].
'</a>';
619 $html .= $product[
'author'];
623 $html .=
'<br><span class="small">';
628 $html .=
'<small class="appDateCreation appRef"> ';
629 if (empty($product[
'tms'])) {
630 $html .=
img_picto($langs->trans(
'DateCreation'),
'calendar',
'class="pictofixedwidth"').
'<span class="opacitymedium"><span class="hideonsmartphone">'.$langs->trans(
"DateCreation").
': </span>';
633 $html .=
img_picto($langs->trans(
'DateModification'),
'calendar',
'class="pictofixedwidth"').
'<span class="opacitymedium">'.
dol_print_date(
dol_stringtotime($product[
'tms']),
'day').
'</span>';
635 $html .=
' ';
637 $html .=
'<div class="appSource inline-block valigntop">';
638 if ($product[
"source"] ==
'dolistore') {
639 $html .=
'<img border="0" title="'.dolPrintHTML($langs->trans(
'Source').
": DoliStore").
'" class="imgautosize valignmiddle inline-block pictofixedwidth" style="height: 14px" src="'.DOL_URL_ROOT.
'/theme/dolistore_squarred.svg">';
640 } elseif ($product[
"source"] ==
'githubcommunity') {
641 $html .=
img_picto($langs->trans(
'Source').
': GitHub community repo',
'group',
'class="pictofixedwidth valignmiddle"');
643 $html .=
img_picto($langs->trans(
'Source').
': '.$langs->trans(
'Other'),
'generic',
'class="pictofixedwidth"');
647 $html .= $langs->trans(
'Ref').
' '.
dolPrintHTML(preg_replace(
'/@.*$/',
'', $product[
"ref"]));
648 $html .=
'</small><br>';
652 if (!empty($product[
'phpmin']) && $product[
'phpmin'] !=
'unknown') {
653 $html .=
' <span class="badge-secondary small" style="padding: 3px; border-radius: 5px">PHP min '.$product[
'phpmin'].
'</span>';
655 if (!empty($product[
'phpmax']) && $product[
'phpmax'] !=
'unknown') {
656 $html .=
' <span class="badge-secondary small" style="padding: 3px; border-radius: 5px">PHP max '.$product[
'phpmax'].
'</span>';
661 $html .=
'<div class="storedesc">'.dolPrintHTML(
dol_string_nohtmltag($product[
"description"])).
'</div>';
665 $html .=
'</tr><tr class="app2 oddeven nohover borderbottom '.dol_escape_htmltag($compatible).
'">';
669 $html .=
'<td class="margeCote center amount'.(getDolOptimizeSmallScreen() ?
' left" colspan="2"' :
'"').
'>';
671 if (($product[
'direct-download'] && $product[
'direct-download'] ==
'yes')
672 || ($product[
'source'] ===
'dolistore' && empty((
float) $product[
'price_ht']))) {
674 $html .= $install_link;
680 $html .=
'<td class="margeCote nowraponall">';
684 $html .= $download_link;
690 if (empty($this->products)) {
692 $langs->load(
"website");
694 $html .=
'<tr class=""><td colspan="'.$colspan.
'" class="center">';
696 $html .= $langs->trans(
"noResultsWereFound").
'...';
698 $html .=
'</td></tr>';
705 $(document).on("click","[data-confirm]",function(){
706 var button = $(this);
707 var confirmTitle = button.data("confirm-title");
708 var confirmText = button.data("confirm-text");
710 buttons[button.data("confirm-label")||"' . $confirmLabel .
'"] = function(){
711 var form = $("<form method=\'POST\' style=\'display:none\'>").attr("action", button.data("url"));
712 $.each(button.data("fields"), function(name, value){
713 form.append($("<input type=\'hidden\'>").attr("name", name).val(value));
715 $("body").append(form);
717 $(this).dialog("close");
719 buttons["' . $cancelLabel .
'"] = function(){$(this).dialog("close");};
720 $("<div>").html(confirmText).dialog({
729 $this->numberOfProducts = count($this->products);
748 function (array $a, array $b) use ($key) {
749 $valA = isset($a[$key]) && is_scalar($a[$key]) ? (
string) $a[$key] :
'';
750 $valB = isset($b[$key]) && is_scalar($b[$key]) ? (
string) $b[$key] :
'';
752 return strnatcmp($valA, $valB);
766 $v1 = str_replace(array(
'v',
'V'),
'', $v1);
767 $v2 = str_replace(array(
'v',
'V'),
'', $v2);
769 $v1 = explode(
'.', $v1);
770 $v2 = explode(
'.', $v2);
773 $count1 = count($v1);
774 $count2 = count($v2);
775 $maxcount = max($count1, $count2);
776 while ($level < $maxcount) {
777 $operande1 = isset($v1[$level]) ? $v1[$level] :
'x';
778 $operande2 = isset($v2[$level]) ? $v2[$level] :
'x';
780 if (strtoupper($operande1) ==
'X' || strtoupper($operande2) ==
'X' || $operande1 ==
'*' || $operande2 ==
'*') {
783 if ($operande1 < $operande2) {
787 if ($operande1 > $operande2) {
806 return '<a href="'.$this->get_previous_url().
'" class="button">'.
dol_escape_htmltag($text).
'</a>';
819 return '<a href="'.$this->get_next_url().
'" class="button">'.
dol_escape_htmltag($text).
'</a>';
831 $param_array = array();
832 if ($this->no_page > 1) {
837 if (!empty($this->search)) {
838 $param_array[
'search_keyword'] = $this->search;
840 $param_array[
'no_page'] = $this->no_page - $sub;
841 if ($this->categorie != 0) {
842 $param_array[
'categorie'] = $this->categorie;
844 $param = http_build_query($param_array);
845 return $this->url.
"&".$param;
857 $param_array = array();
858 if ($this->products !==
null && count($this->products) < $this->per_page) {
863 if (!empty($this->search)) {
864 $param_array[
'search_keyword'] = $this->search;
866 $param_array[
'no_page'] = $this->no_page + $add;
867 if ($this->categorie != 0) {
868 $param_array[
'categorie'] = $this->categorie;
870 $param = http_build_query($param_array);
871 return $this->url.
"&".$param;
884 $page = $this->no_page;
885 $limit = $this->per_page;
886 $totalnboflines = $this->numberTotalOfProducts ?: 0;
887 $num = $this->numberOfProducts;
893 if ($page > 0 || $num > $limit) {
894 if ($totalnboflines) {
896 $nbpages = $this->numberTotalOfPages;
903 $pagelist .=
'<li class="pagination paginationpage paginationpageleft"><a class="paginationprevious reposition" href="'.$this->get_previous_url().
'"><i class="fa fa-chevron-left" title="'.
dol_escape_htmltag($langs->trans(
"Previous")).
'"></i></a></li>';
906 $pagelist .=
'<li class="pagination">';
907 $pagelist .=
'<label for="page_input">Page </label>';
908 if ($this->categorie != 0) {
909 $pagelist .=
'<input type="hidden" name="categorie" value="' . $this->categorie .
'">';
911 $pagelist .=
'<input type="text" id="page_input" name="no_page" value="'.($page).
'" min="1" max="'.$nbpages.
'" class="width40 page_input right" oninput="if(this.value > '.$nbpages.
') this.value='.$nbpages.
'">';
912 $pagelist .=
' / '.$nbpages;
913 $pagelist .=
'</li>';
916 if ($page < $nbpages) {
917 $pagelist .=
'<li class="pagination paginationpage paginationpageright"><a class="paginationnext reposition" href="'.$this->get_next_url().
'"><i class="fa fa-chevron-right" title="'.
dol_escape_htmltag($langs->trans(
"Next")).
'"></i></a></li>';
922 if ($limit || $pagelist) {
923 $html .=
'<div class="pagination" style="padding: 7px;">';
930 $html .= ajax_autoselect(
'.page_input');
946 400 =>
'Bad Request',
947 401 =>
'Unauthorized',
949 405 =>
'Method Not Allowed',
950 500 =>
'Internal Server Error',
954 if ($request[
'status_code'] === 200 || $request[
'status_code'] === 201) {
959 $error_message = $error_messages[$request[
'status_code']] ??
'Unexpected HTTP status: ' . $request[
'status_code'];
962 if (!empty($request[
'response']) && isset($request[
'response'][
'errors']) && is_array($request[
'response'][
'errors'])) {
963 foreach ($request[
'response'][
'errors'] as $error) {
964 $error_message .=
' - (Code ' . $error[
'code'] .
'): ' . $error[
'message'];
968 if (!empty($request[
'curl_error_msg'])) {
969 $error_message .=
' - ' . $request[
'curl_error_msg'];
973 return sprintf(
'This call to the API failed and returned an HTTP status of %d. That means: %s.', $request[
'status_code'], $error_message);
986 $cache_file = $this->cache_file;
987 $cache_folder = dirname($cache_file);
994 if (!file_exists($cache_file) || filemtime($cache_file) < (
dol_now() - $cache_time)) {
996 $addheaders = array();
997 $result =
getURLContent($file_source_url,
'GET',
'', 1, $addheaders);
998 if (!empty($result) && $result[
'http_code'] == 200) {
999 $yaml = $result[
'content'];
1000 $result = file_put_contents($cache_file, $yaml);
1001 if ($result ===
false) {
1002 $this->error =
'Failed to create cache file: ' . $cache_file;
1008 $yaml = file_get_contents($cache_file);
1024 $currentPackage =
null;
1025 $currentSection =
null;
1027 foreach (explode(
"\n", trim($yaml)) as $line) {
1028 $trimmedLine = trim($line);
1031 if ($trimmedLine ===
'' || strpos($trimmedLine,
'#') === 0) {
1037 if (preg_match(
'/^\s*-\s*modulename:\s*["\']?(.*?)["\']?$/', $trimmedLine, $matches)) {
1038 if ($currentPackage !==
null) {
1040 if (!empty($currentPackage[
'status']) && in_array($currentPackage[
'status'], array(
'enabled',
'soon'))) {
1041 $data[] = $currentPackage;
1044 $currentPackage = [
'modulename' => $matches[1]];
1045 $currentSection =
null;
1050 if (!preg_match(
'/^\s*(fr|en|es|it|de):\s*["\']?(.*?)["\']?$/', $trimmedLine)) {
1051 $currentSection =
null;
1055 if (preg_match(
'/^(\w[\w-]*):\s*["\']?(.*?)["\']?$/', $trimmedLine, $matches)) {
1056 if ($currentPackage !==
null) {
1057 if ($currentSection) {
1059 $currentPackage[$currentSection][$matches[1]] = $matches[2] ===
'' ? null : $matches[2];
1062 $currentPackage[$matches[1]] = $matches[2] ===
'' ? null : $matches[2];
1067 if (preg_match(
'/^\s*(label|description):\s*$/', $trimmedLine, $matches)) {
1068 $currentSection = $matches[1];
1069 $currentPackage[$currentSection] = [];
1077 if ($currentPackage !==
null) {
1078 if (!empty($currentPackage[
'status']) && in_array($currentPackage[
'status'], array(
'enabled',
'soon'))) {
1079 $data[] = $currentPackage;
1097 if (!is_array($data) || empty($data) || empty($source)) {
1098 return $adaptedData;
1101 if ($source ===
'githubcommunity') {
1102 foreach ($data as $package) {
1103 if (empty($package[
'modulename'])) {
1110 if (!empty($package[
'dolistore-download']) && preg_match(
'/www\.dolistore\.com\/product\.php\?id=(\d+)/', (
string) $package[
'dolistore-download'], $reg)) {
1116 'ref' => str_replace(
' ',
'', $package[
'modulename'] .
'-' . $package[
'current_version'] .
'@' .
1117 (array_key_exists(
'author', $package) ? $package[
'author'] :
'unkownauthor')),
1118 'label' => !empty($package[
'label'][substr($this->lang, 0, 2)])
1119 ? $package[
'label'][substr($this->lang, 0, 2)]
1120 : (!empty($package[
'label'][
'en']) ? $package[
'label'][
'en'] : $package[
'modulename']),
1121 'description' => !empty($package[
'description'][substr($this->lang, 0, 2)])
1122 ? $package[
'description'][substr($this->lang, 0, 2)]
1123 : (!empty($package[
'description'][
'en']) ? $package[
'description'][
'en'] :
''),
1124 'datec' => (!empty($package[
'created_at']) && is_string($package[
'created_at']))
1125 ?
date(
'Y-m-d H:i:s', strtotime($package[
'created_at']))
1127 'tms' => (!empty($package[
'last_updated_at']) && is_string($package[
'last_updated_at']))
1128 ?
date(
'Y-m-d H:i:s', strtotime($package[
'last_updated_at']))
1130 'author' => array_key_exists(
'author', $package) ? $package[
'author'] :
'',
1131 'author_url' => array_key_exists(
'author_url', $package) ? $package[
'author_url'] :
'',
1132 'dolibarr_min' => !empty($package[
'dolibarrmin'])
1133 ? $package[
'dolibarrmin']
1135 'dolibarr_max' => !empty($package[
'dolibarrmax'])
1136 ? $package[
'dolibarrmax']
1138 'phpmin' => !empty($package[
'phpmin'])
1139 ? $package[
'phpmin']
1141 'phpmax' => !empty($package[
'phpmax'])
1142 ? $package[
'phpmax']
1144 'module_version' => !empty($package[
'current_version'])
1145 ? $package[
'current_version']
1147 'cover_photo_url' => !empty($package[
'cover'])
1150 'category' => (!empty($package[
'category']) && is_string($package[
'category']))
1151 ? explode(
',', str_replace(
' ',
'', (
string) $package[
'category']))
1153 'link' => !empty($package[
'git'])
1156 'source' =>
'githubcommunity',
1157 'status' => !empty($package[
'status']) ? $package[
'status'] :
'',
1158 'direct-download' => !empty($package[
'direct-download'])
1159 ? $package[
'direct-download']
1161 'dolistore-download' => !empty($package[
'dolistore-download'])
1162 ? $package[
'dolistore-download']
1167 if (array_key_exists(
'price', $package) && $package[
'price'] !=
null) {
1168 $adaptedPackage[
'price_ht'] = $package[
'price'];
1171 $adaptedData[] = $adaptedPackage;
1175 if ($source ===
'dolistore') {
1176 foreach ($data as $package) {
1177 $urlphoto = $this->shop_url.$package[
'cover_photo_url'];
1179 if (preg_match(
'/^\/?wrapper\.php\?hashp=/', $package[
'cover_photo_url']) && !preg_match(
'/attachment=/', $package[
'cover_photo_url'])) {
1180 $urlphoto .=
'&attachment=0';
1184 'id' => $package[
'id'],
1185 'ref' => $package[
'ref'],
1186 'label' => $package[
'label'],
1187 'description' => $package[
'description'],
1188 'datec' => $package[
'datec'],
1189 'tms' => $package[
'tms'],
1190 'author' => array_key_exists(
'author', $package) ? $package[
'author'] :
'',
1191 'author_url' => array_key_exists(
'author_url', $package) ? $package[
'author_url'] :
'',
1192 'price_ttc' => $package[
'price_ttc'],
1193 'price_ht' => $package[
'price_ht'],
1194 'dolibarr_min' => $package[
'dolibarr_min'],
1195 'dolibarr_max' => $package[
'dolibarr_max'],
1196 'phpmin' => empty($package[
'phpmin']) ?
'' : $package[
'phpmin'],
1197 'phpmax' => empty($package[
'phpmax']) ?
'' : $package[
'phpmax'],
1198 'module_version' => $package[
'module_version'],
1199 'cover_photo_url' => $urlphoto,
1200 'source' =>
'dolistore',
1201 'status' => empty($package[
'status']) ?
'' : $package[
'status']
1204 $adaptedData[$package[
'id']] = $adaptedPackage;
1208 return $adaptedData;
1220 $filteredData = $list;
1232 static function ($a, $b) {
1233 return strtotime($b[
'datec'] ??
'0') - strtotime($a[
'datec'] ??
'0');
1237 if (!empty($options[
'search'])) {
1238 $filteredData = array_filter(
1247 static function ($package) use ($options) {
1248 return stripos($package[
'label'], $options[
'search']) !==
false || stripos($package[
'description'], $options[
'search']) !==
false;
1253 if (!empty($options[
'categorieid'])) {
1254 $filteredData = array_filter(
1263 static function ($package) use ($options) {
1264 return in_array($options[
'categorieid'], $package[
'category']);
1269 $total = count($filteredData);
1272 $filteredData = array_values($filteredData);
1273 $filteredData = array_slice($filteredData, ($options[
'page'] - 1) * $options[
'limit'], $options[
'limit']);
1275 return [
'total' => $total,
'data' => $filteredData];
1286 $testRequest = $this->
callApi(
'categories');
1288 if (!isset($testRequest[
'response']) || !is_array($testRequest[
'response']) || ($testRequest[
'status_code'] != 200 && $testRequest[
'status_code'] != 201)) {
1304 public function libStatus($status, $mode = 3, $moretext =
'')
1308 $statusType =
'status4';
1310 $statusType =
'status8';
1314 $labelStatusShort = [];
1316 $labelStatus[0] = $langs->transnoentitiesnoconv(
"NotConnected");
1317 $labelStatus[1] = $langs->transnoentitiesnoconv(
"online");
1318 $labelStatusShort[0] = $langs->transnoentitiesnoconv(
"NotConnected");
1319 $labelStatusShort[1] = $langs->transnoentitiesnoconv(
"online");
1321 return dolGetStatus($labelStatus[$status], $labelStatusShort[$status],
'', $statusType, $mode,
'', array(
'badgeParams' => array(
'attr' => array(
'class' =>
'classfortooltip',
'title' => $labelStatusShort[$status].$moretext))));
1336 if (!function_exists(
'curl_init')) {
1337 dol_syslog(__METHOD__ .
': cURL is not available', LOG_ERR);
1342 if (empty($producttoinstall[
'ref'])) {
1343 dol_syslog(__METHOD__ .
': Missing producttoinstall', LOG_ERR);
1347 $current_version = $producttoinstall[
'module_version'] ??
'';
1348 $module_name = strtolower(preg_replace(
'/@.*$/',
'', $producttoinstall[
'ref'] ??
''));
1351 $module_name = preg_replace(
'/-' . preg_quote($current_version,
'/') .
'$/',
'', $module_name);
1353 if (empty($module_name) || empty($current_version) || $current_version ==
'unknown') {
1354 dol_syslog(__METHOD__ .
': Missing or unknown module name/version for product', LOG_ERR);
1359 $tmpdir =
$conf->admin->dir_temp .
'/remotestoredl';
1362 $downloaded =
false;
1363 switch ($producttoinstall[
'source']) {
1365 if ($producttoinstall[
'id'] > 0) {
1366 $source_url =
'https://www.dolistore.com/_service_download.php?t=free&p=' . $producttoinstall[
'id'];
1369 dol_syslog(__METHOD__ .
': Dolistore download failed: ' . $source_url, LOG_ERR);
1373 dol_syslog(__METHOD__ .
': Invalid product ID for Dolistore download: ' . $producttoinstall[
'id'], LOG_ERR);
1377 case 'githubcommunity':
1378 if ($producttoinstall[
'direct-download'] && $producttoinstall[
'direct-download'] ==
'yes') {
1379 $source_url =
'https://github.com/Dolibarr/dolibarr-community-modules/raw/refs/heads/main/' . $module_name .
'/module_' . $module_name .
'-' . $current_version .
'.zip';
1382 dol_syslog(__METHOD__ .
': GitHub community module download failed: ' . $source_url .
', Try to find a Dolistore link', LOG_WARNING);
1383 if ($producttoinstall[
'id'] > 0) {
1384 $source_url =
'https://www.dolistore.com/_service_download.php?t=free&p=' . $producttoinstall[
'id'];
1387 dol_syslog(__METHOD__ .
': Dolistore download failed: ' . $source_url, LOG_ERR);
1391 dol_syslog(__METHOD__ .
': No direct download available for this GitHub community module', LOG_ERR);
1396 dol_syslog(__METHOD__ .
': No direct download available for this GitHub community module', LOG_ERR);
1401 dol_syslog(__METHOD__ .
': Unsupported source type: ' . $producttoinstall[
'source'], LOG_ERR);
1405 dol_syslog(__METHOD__ .
': Module downloaded successfully to: ' . $downloaded, LOG_DEBUG);
1423 if (!empty($head[
'header'])) {
1424 if (preg_match_all(
'/Content-Disposition:.*filename=["\']?([^"\';\r\n]+)/i', $head[
'header'], $m)) {
1425 $filename = trim(end($m[1]),
" \t\"'");
1430 if (empty($filename)) {
1431 $filename = basename(parse_url($url, PHP_URL_PATH));
1435 if (empty($filename) || !preg_match(
'/^module_[a-z0-9_]+-[0-9]+\.[0-9]+\.[0-9]+\.zip$/i', $filename)) {
1436 dol_syslog(__METHOD__ .
': Cannot determine filename from URL: ' . $url, LOG_ERR);
1442 if (empty($response[
'content']) || (isset($response[
'http_code']) && $response[
'http_code'] !== 200)) {
1444 __METHOD__ .
': Download failed — HTTP ' . ($response[
'http_code'] ??
'unknown') .
' — ' . $url,
1451 $dest_file = $dest_path .
'/' . $filename;
1456 $writtenfile = file_put_contents(
dol_osencode($dest_file), $response[
'content']);
1457 if ($writtenfile ===
false || $writtenfile === 0) {
1458 dol_syslog(__METHOD__ .
': Cannot write file: ' . $dest_file, LOG_ERR);
1463 dol_syslog(__METHOD__ .
': Downloaded successfully to: ' . $dest_file, LOG_DEBUG);
$id
Support class for third parties, contacts, members, users or resources.
__construct($debug=false)
Constructor.
libStatus($status, $mode=3, $moretext='')
Retrieve the status icon.
checkApiStatus()
Check if an Dolistore API is up.
getProducts($options)
Generate HTML for products.
versionCompare($v1, $v2)
version compare
getPagination()
Generate pagination for navigating through pages of products.
get_next_url()
get next url
getRemoteYamlFile($file_source_url, $cache_time)
Get YAML file from remote source and put it into the cache file.
get_previous_link($text='<<')
get previous link
buildSorter(string $key)
Sort an array by a key.
getModuleZIP($producttoinstall=array())
Download a Dolibarr module from a Git repository URL or Dolistore download URL.
getCategories($active=0)
Generate HTML for categories and their children.
applyFilters($list, $options)
Apply filters to the data.
get_next_link($text='> >')
get next link
fetchModulesFromFile($options=array())
Fetch modules from a cache YAML file.
checkStatusCode($request)
Check the status code of the request.
_downloadFile(string $url, string $dest_path)
Download a remote URL to a local file using getURLContent (native Dolibarr).
readYaml($yaml)
Read a YAML string and convert it to an array.
callApi($resource, $options=false)
Test if we can access to remote Dolistore market place.
adaptData($data, $source)
Adapter data fetched from github remote source to the expected format.
loadRemoteSources($debug=false)
loadRemoteSources
get_previous_url()
get previous url
dol_stringtotime($string, $gm=1)
Convert a string date into a GM Timestamps date Warning: YYYY-MM-DDTHH:MM:SS+02:00 (RFC3339) is not s...
dol_is_file($pathoffile)
Return if path is a file.
dol_is_dir($folder)
Test if filename is a directory.
dol_now($mode='gmt')
Return date for now.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2, $allowothertags=array())
Show picto whatever it's its name (generic function)
dolPrintHTML($s, $allowiframe=0, $moreallowedtags=array())
Return a string (that can be on several lines) ready to be output on a HTML page.
dol_osencode($str)
Return a string encoded into OS filesystem encoding.
dol_string_nohtmltag($stringtoclean, $removelinefeed=1, $pagecodeto='UTF-8', $strip_tags=0, $removedoublespaces=1)
Clean a string from all HTML tags and entities.
price2num($amount, $rounding='', $option=0)
Function that return a number with universal decimal format (decimal separator is '.
getDolOptimizeSmallScreen()
Return if render must be optimized for small screen.
dolChmod($filepath, $newmask='')
Change mod of a file.
getDolGlobalInt($key, $default=0)
Return a Dolibarr global constant int value.
dol_escape_js($stringtoescape, $mode=0, $noescapebackslashn=0)
Returns text escaped for inclusion into JavaScript code.
dol_print_date($time, $format='', $tzoutput='auto', $outputlangs=null, $encodetooutput=false, $decorate=0)
Output date in a string format according to outputlangs (or langs if not defined).
getDolGlobalString($key, $default='')
Return a Dolibarr global constant string value.
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.
dol_mkdir($dir, $dataroot='', $newmask='')
Creation of a directory (this can create recursive subdir)
dol_escape_htmltag($stringtoescape, $keepb=0, $keepn=0, $noescapetags='', $escapeonlyhtmltags=0, $cleanalsojavascript=0)
Returns text escaped for inclusion in HTML alt or title or value tags, or into values of HTML input f...
getURLContent($url, $postorget='GET', $param='', $followlocation=1, $addheaders=array(), $allowedschemes=array('http', 'https'), $localurl=0, $ssl_verifypeer=-1, $timeoutconnect=0, $timeoutresponse=0, $otherCurlOptions=array(), $morelogsuffix='')
Function to get a content from an URL (use proxy if proxy defined).
print $langs trans("Show") . '< td style="' . $timeColor . '" align="center"> s</td > badge status0 badge status4 badge status3 Error badge status8< td align="center">< span class="badge ' . $badge . '"></span ></td >< td align="center">< a href="#" class="button button-small" onclick="openLogModal(this)" data-req="' . dol_escape_htmltag($reqSafe) . '" data-res="' . dol_escape_htmltag($resSafe) . '" data-err="' . dol_escape_htmltag($errSafe) . '">< span class="fa fa-search-plus"></span ></a ></td ></tr >< tr >< td colspan="' . $colspan . '" class="opacitymedium"></td ></tr ></table ></div ></form > logModal none logModal none s a JSON string
buildzip.php
print $langs trans('Date')." left Ref Label right Qty right Price right TotalHT right TotalTTC right right right right right right right right right centpercent right TotalHT right n right VAT right n right TotalVAT right n No sujeto a RE IRPF right TotalLT1 right n right TotalLT2 right n right TotalTTC right n takeposcustomercurrency takeposcustomercurrency takeposcustomercurrency takeposcustomercurrency right TotalTTC takeposcustomercurrency right takeposcustomercurrency n right Paid right PaymentTypeShortLIQ right SELECT p pos_change as p datep as date