Files
2025-06-17 11:53:18 +02:00

237 lines
7.9 KiB
PHP

<?php
/**
* @package Joomla.Site
* @subpackage com_ajax
*
* @copyright (C) 2013 Open Source Matters, Inc. <https://www.joomla.org>
* @license GNU General Public License version 2 or later; see LICENSE.txt
*/
\defined('_JEXEC') or die;
use Joomla\CMS\Event\Plugin\AjaxEvent;
use Joomla\CMS\Factory;
use Joomla\CMS\Language\Text;
use Joomla\CMS\Log\Log;
use Joomla\CMS\Plugin\PluginHelper;
use Joomla\CMS\Response\JsonResponse;
use Joomla\CMS\Table\Table;
/*
* References
* Support plugins in your component
* - https://docs.joomla.org/Special:MyLanguage/Supporting_plugins_in_your_component
*
* Best way for JSON output
* - https://groups.google.com/d/msg/joomla-dev-cms/WsC0nA9Fixo/Ur-gPqpqh-EJ
*/
/** @var \Joomla\CMS\Application\CMSApplication $app */
$app = Factory::getApplication();
$app->allowCache(false);
// Prevent the api url from being indexed
$app->setHeader('X-Robots-Tag', 'noindex, nofollow');
// JInput object
$input = $app->getInput();
// Requested format passed via URL
$format = strtolower($input->getWord('format', ''));
// Initialize default response and module name
$results = null;
$parts = null;
// Check for valid format
if (!$format) {
$results = new InvalidArgumentException(Text::_('COM_AJAX_SPECIFY_FORMAT'), 404);
} elseif ($input->get('module')) {
/**
* Module support.
*
* modFooHelper::getAjax() is called where 'foo' is the value
* of the 'module' variable passed via the URL
* (i.e. index.php?option=com_ajax&module=foo).
*
*/
$module = $input->get('module');
$table = Table::getInstance('extension');
$moduleId = $table->find(['type' => 'module', 'element' => 'mod_' . $module]);
if ($moduleId && $table->load($moduleId) && $table->enabled) {
$helperFile = JPATH_BASE . '/modules/mod_' . $module . '/helper.php';
if (strpos($module, '_')) {
$parts = explode('_', $module);
} elseif (strpos($module, '-')) {
$parts = explode('-', $module);
}
if ($parts) {
$class = 'Mod';
foreach ($parts as $part) {
$class .= ucfirst($part);
}
$class .= 'Helper';
} else {
$class = 'Mod' . ucfirst($module) . 'Helper';
}
$method = $input->get('method') ?: 'get';
$moduleInstance = $app->bootModule('mod_' . $module, $app->getName());
if ($moduleInstance instanceof \Joomla\CMS\Helper\HelperFactoryInterface && $helper = $moduleInstance->getHelper(substr($class, 3))) {
$results = method_exists($helper, $method . 'Ajax') ? $helper->{$method . 'Ajax'}() : null;
}
if ($results === null && is_file($helperFile)) {
JLoader::register($class, $helperFile);
if (method_exists($class, $method . 'Ajax')) {
// Load language file for module
$basePath = JPATH_BASE;
$lang = Factory::getLanguage();
$lang->load('mod_' . $module, $basePath)
|| $lang->load('mod_' . $module, $basePath . '/modules/mod_' . $module);
try {
$results = \call_user_func($class . '::' . $method . 'Ajax');
} catch (Exception $e) {
$results = $e;
}
} else {
// Method does not exist
$results = new LogicException(Text::sprintf('COM_AJAX_METHOD_NOT_EXISTS', $method . 'Ajax'), 404);
}
} elseif ($results === null) {
// The helper file does not exist
$results = new RuntimeException(Text::sprintf('COM_AJAX_FILE_NOT_EXISTS', 'mod_' . $module . '/helper.php'), 404);
}
} else {
// Module is not published, you do not have access to it, or it is not assigned to the current menu item
$results = new LogicException(Text::sprintf('COM_AJAX_MODULE_NOT_ACCESSIBLE', 'mod_' . $module), 404);
}
} elseif ($input->get('plugin')) {
/**
* Plugin support by default is based on the "Ajax" plugin group.
* An optional 'group' variable can be passed via the URL.
*
* The plugin event triggered is onAjaxFoo, where 'foo' is
* the value of the 'plugin' variable passed via the URL
* (i.e. index.php?option=com_ajax&plugin=foo)
*
*/
try {
$dispatcher = $app->getDispatcher();
$group = $input->get('group', 'ajax');
$eventName = 'onAjax' . ucfirst($input->get('plugin', ''));
PluginHelper::importPlugin($group, null, true, $dispatcher);
$results = $dispatcher->dispatch($eventName, new AjaxEvent($eventName, ['subject' => $app]))->getArgument('result', []);
} catch (Throwable $e) {
$results = $e;
}
} elseif ($input->get('template')) {
/**
* Template support.
*
* tplFooHelper::getAjax() is called where 'foo' is the value
* of the 'template' variable passed via the URL
* (i.e. index.php?option=com_ajax&template=foo).
*
*/
$template = $input->get('template');
$table = Table::getInstance('extension');
$templateId = $table->find(['type' => 'template', 'element' => $template]);
if ($templateId && $table->load($templateId) && $table->enabled) {
$basePath = ($table->client_id) ? JPATH_ADMINISTRATOR : JPATH_SITE;
$helperFile = $basePath . '/templates/' . $template . '/helper.php';
if (strpos($template, '_')) {
$parts = explode('_', $template);
} elseif (strpos($template, '-')) {
$parts = explode('-', $template);
}
if ($parts) {
$class = 'Tpl';
foreach ($parts as $part) {
$class .= ucfirst($part);
}
$class .= 'Helper';
} else {
$class = 'Tpl' . ucfirst($template) . 'Helper';
}
$method = $input->get('method') ?: 'get';
if (is_file($helperFile)) {
JLoader::register($class, $helperFile);
if (method_exists($class, $method . 'Ajax')) {
// Load language file for template
$lang = Factory::getLanguage();
$lang->load('tpl_' . $template, $basePath)
|| $lang->load('tpl_' . $template, $basePath . '/templates/' . $template);
try {
$results = \call_user_func($class . '::' . $method . 'Ajax');
} catch (Exception $e) {
$results = $e;
}
} else {
// Method does not exist
$results = new LogicException(Text::sprintf('COM_AJAX_METHOD_NOT_EXISTS', $method . 'Ajax'), 404);
}
} else {
// The helper file does not exist
$results = new RuntimeException(Text::sprintf('COM_AJAX_FILE_NOT_EXISTS', 'tpl_' . $template . '/helper.php'), 404);
}
} else {
// Template is not assigned to the current menu item
$results = new LogicException(Text::sprintf('COM_AJAX_TEMPLATE_NOT_ACCESSIBLE', 'tpl_' . $template), 404);
}
}
// Return the results in the desired format
switch ($format) {
// JSONinzed
case 'json':
echo new JsonResponse($results, null, false, $input->get('ignoreMessages', true, 'bool'));
break;
// Handle as raw format
default:
// Output exception
if ($results instanceof Exception) {
// Log an error
Log::add($results->getMessage(), Log::ERROR);
// Set status header code
$app->setHeader('status', $results->getCode(), true);
// Echo exception type and message
$out = \get_class($results) . ': ' . $results->getMessage();
} elseif (is_scalar($results)) {
// Output string/ null
$out = (string) $results;
} else {
// Output array/ object
$out = implode((array) $results);
}
echo $out;
break;
}