salesagility_SuiteCRM/modules/Connectors/controller.php

824 lines
30 KiB
PHP
Executable File

<?php
if (!defined('sugarEntry') || !sugarEntry) {
die('Not A Valid Entry Point');
}
/**
*
* SugarCRM Community Edition is a customer relationship management program developed by
* SugarCRM, Inc. Copyright (C) 2004-2013 SugarCRM Inc.
*
* SuiteCRM is an extension to SugarCRM Community Edition developed by SalesAgility Ltd.
* Copyright (C) 2011 - 2018 SalesAgility Ltd.
*
* This program is free software; you can redistribute it and/or modify it under
* the terms of the GNU Affero General Public License version 3 as published by the
* Free Software Foundation with the addition of the following permission added
* to Section 15 as permitted in Section 7(a): FOR ANY PART OF THE COVERED WORK
* IN WHICH THE COPYRIGHT IS OWNED BY SUGARCRM, SUGARCRM DISCLAIMS THE WARRANTY
* OF NON INFRINGEMENT OF THIRD PARTY RIGHTS.
*
* 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 Affero General Public License for more
* details.
*
* You should have received a copy of the GNU Affero General Public License along with
* this program; if not, see http://www.gnu.org/licenses or write to the Free
* Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301 USA.
*
* You can contact SugarCRM, Inc. headquarters at 10050 North Wolfe Road,
* SW2-130, Cupertino, CA 95014, USA. or at email address contact@sugarcrm.com.
*
* The interactive user interfaces in modified source and object code versions
* of this program must display Appropriate Legal Notices, as required under
* Section 5 of the GNU Affero General Public License version 3.
*
* In accordance with Section 7(b) of the GNU Affero General Public License version 3,
* these Appropriate Legal Notices must retain the display of the "Powered by
* SugarCRM" logo and "Supercharged by SuiteCRM" logo. If the display of the logos is not
* reasonably feasible for technical reasons, the Appropriate Legal Notices must
* display the words "Powered by SugarCRM" and "Supercharged by SuiteCRM".
*/
require_once('include/connectors/sources/SourceFactory.php');
require_once('include/connectors/ConnectorFactory.php');
require_once('include/MVC/Controller/SugarController.php');
#[\AllowDynamicProperties]
class ConnectorsController extends SugarController
{
public $admin_actions = array('ConnectorSettings', 'DisplayProperties', 'MappingProperties', 'ModifyMapping', 'ModifyDisplay', 'ModifyProperties',
'ModifySearch', 'SearchProperties', 'SourceProperties',
'SavedModifyDisplay', 'SaveModifyProperties', 'SaveModifySearch', 'RunTest');
public function process()
{
if (!is_admin($GLOBALS['current_user']) && in_array($this->action, $this->admin_actions)) {
$this->hasAccess = false;
}
parent::process();
}
/**
* When the user clicks the Search button, the form is posted back here and this action sets the
* search parameters in the session. Once this call returns, the tabs will then call RetrieveSource to load
* the data that was saved in the session.
*
*/
public function action_SetSearch()
{
if (empty($_REQUEST)) {
return;
}
$this->view = 'ajax';
require_once('include/connectors/utils/ConnectorUtils.php');
$searchdefs = ConnectorUtils::getSearchDefs();
$merge_module = $_REQUEST['merge_module'];
$record_id = $_REQUEST['record'];
$searchDefs = isset($searchdefs) ? $searchdefs : array();
unset($_SESSION['searchDefs'][$merge_module][$record_id]);
$sMap = array();
$search_source = $_REQUEST['source_id'];
$source_instance = ConnectorFactory::getInstance($search_source);
$source_map = $source_instance->getModuleMapping($merge_module);
$module_fields = array();
foreach ($_REQUEST as $search_term => $val) {
if (!empty($source_map[$search_term])) {
$module_fields[$source_map[$search_term]] = $val;
}
}
foreach ($module_fields as $search_term => $val) {
foreach ($searchDefs as $source => $modules) {
if (empty($sMap[$source])) {
$instance = ConnectorFactory::getInstance($source);
$sMap[$source] = array_flip($instance->getModuleMapping($merge_module));
}
if (!empty($sMap[$source][$search_term])) {
$source_key = $sMap[$source][$search_term];
$_SESSION['searchDefs'][$merge_module][$record_id][$source][$source_key] = $val;
}
}
}
}
/**
* This action it meant to handle the hover action on the listview.
*
*/
public function action_RetrieveSourceDetails()
{
$results = [];
$this->view = 'ajax';
$source_id = $_REQUEST['source_id'];
$record_id = $_REQUEST['record_id'];
if (empty($source_id) || empty($record_id)) {
//display error here
return;
}
$source = ConnectorFactory::getInstance($source_id);
$module = $_SESSION['merge_module'];
$result = $source->fillBean(array('id' => $record_id), $module);
require_once('include/connectors/utils/ConnectorUtils.php');
$connector_strings = ConnectorUtils::getConnectorStrings($source_id);
$fields = $source->getModuleMapping($module);
$fieldDefs = $source->getFieldDefs();
$str = '';
foreach ($fields as $key=>$field) {
$label = $field;
if (isset($fieldDefs[$key])) {
$label = isset($connector_strings[$fieldDefs[$key]['vname']]) ? $connector_strings[$fieldDefs[$key]['vname']] : $label;
}
$val = $result->$field;
if (!empty($val)) {
if (strlen((string) $val) > 50) {
$val = substr((string) $val, 0, 47) . '...';
}
$str .= $label . ': ' . $val.'<br/>';
}
}
global $theme;
$json = getJSONobj();
$retArray = array();
$retArray['body'] = !empty($str) ? str_replace(array("\rn", "\r", "\n"), array('','','<br />'), $str) : $GLOBALS['mod_strings']['ERROR_NO_ADDITIONAL_DETAIL'];
$retArray['caption'] = "<div style='float:left'>{$GLOBALS['app_strings']['LBL_ADDITIONAL_DETAILS']}</div>";
$retArray['width'] = (empty($results['width']) ? '300' : $results['width']);
$retArray['theme'] = $theme;
echo 'result = ' . $json->encode($retArray);
}
public function action_GetSearchForm()
{
$this->view = 'ajax';
if (!empty($_REQUEST['source_id'])) {
//get the search fields and return the search form
$ss = new Sugar_Smarty();
require_once('include/connectors/utils/ConnectorUtils.php');
$searchdefs = ConnectorUtils::getSearchDefs();
$merge_module = $_REQUEST['merge_module'];
$seed = loadBean($merge_module);
$_searchDefs = isset($searchdefs) ? $searchdefs : array();
$_trueFields = array();
$source = $_REQUEST['source_id'];
$searchLabels = ConnectorUtils::getConnectorStrings($source);
$record = $_REQUEST['record'];
$sourceObj = SourceFactory::getSource($source);
$field_defs = $sourceObj->getFieldDefs();
if (!empty($_searchDefs[$source][$merge_module])) {
foreach ($_searchDefs[$source][$merge_module] as $key) {
if (!empty($_SESSION['searchDefs'][$merge_module][$record][$source][$key])) {
$_trueFields[$key]['value'] = $_SESSION['searchDefs'][$merge_module][$record][$source][$key];
} else {
$_trueFields[$key]['value'] = '';
}
if (!empty($field_defs[$key]) && isset($searchLabels[$field_defs[$key]['vname']])) {
$_trueFields[$key]['label'] = $searchLabels[$field_defs[$key]['vname']];
} else {
$_trueFields[$key]['label'] = $key;
}
}//foreach
}//fi
$ss->assign('mod', $GLOBALS['mod_strings']);
$ss->assign('search_fields', $_trueFields);
$ss->assign('source_id', $source);
$ss->assign('fields', $seed->field_defs);
$ss->assign('module', $merge_module);
$ss->assign('RECORD', $record);
$ss->assign('APP', $GLOBALS['app_strings']);
$ss->assign('MOD', $GLOBALS['mod_strings']);
echo $ss->fetch('modules/Connectors/tpls/search_form.tpl');
}
}
public function pre_save()
{
}
public function post_save()
{
}
public function action_CallConnectorFunc()
{
$this->view = 'ajax';
$json = getJSONobj();
if (!empty($_REQUEST['source_id'])) {
$source_id = $_REQUEST['source_id'];
require_once('include/connectors/sources/SourceFactory.php');
$source = SourceFactory::getSource($source_id);
$method = 'ext_'.$_REQUEST['source_func'];
if (method_exists($source, $method)) {
echo $json->encode($source->$method($_REQUEST));
} else {
echo $json->encode(array('error'=>true,'errorMessage'=>'Could Not Find Function: '.$method.' in class: '.get_class($source)));
}
} else {
echo $json->encode(array('error'=>true,'errorMessage'=>'Source Id is not specified.'));
}
}
private function remoteFileExists($url)
{
$curl = curl_init($url);
//don't fetch the actual page, you only want to check the connection is ok
curl_setopt($curl, CURLOPT_NOBODY, true);
//do request
$result = curl_exec($curl);
$ret = false;
//if request did not fail
if ($result !== false) {
//if request was ok, check response code
$statusCode = curl_getinfo($curl, CURLINFO_HTTP_CODE);
if ($statusCode == 200) {
$ret = true;
}
}
curl_close($curl);
return $ret;
}
public function action_CallRest()
{
$this->view = 'ajax';
$url = $_REQUEST['url'];
if (!preg_match('/^http[s]{0,1}\:\/\//', (string) $url)) {
throw new RuntimeException('Illegal request');
}
if (!$this->remoteFileExists($url)) {
throw new RuntimeException('Requested URL is not exists.');
}
if (false === ($result = @file_get_contents($_REQUEST['url']))) {
echo '';
} else {
if (!empty($_REQUEST['xml'])) {
$values = array();
$p = xml_parser_create();
xml_parse_into_struct($p, $result, $values);
xml_parser_free($p);
$json = getJSONobj();
echo $json->encode($values);
} else {
echo $result;
}
}
}
public function action_CallSoap()
{
$this->view = 'ajax';
$source_id = $_REQUEST['source_id'];
$module = $_REQUEST['module_id'];
$return_params = explode(',', $_REQUEST['fields']);
require_once('include/connectors/ConnectorFactory.php');
$component = ConnectorFactory::getInstance($source_id);
$beans = $component->fillBeans($_REQUEST, $module);
if (!empty($beans) && !empty($return_params)) {
$results = array();
$count = 0;
foreach ($beans as $bean) {
foreach ($return_params as $field) {
$results[$count][$field] = $bean->$field;
}
$count++;
}
$json = getJSONobj();
echo $json->encode($results);
} else {
echo '';
}
}
public function action_DefaultSoapPopup()
{
$this->view = 'ajax';
$source_id = $_REQUEST['source_id'];
$module = $_REQUEST['module_id'];
$id = $_REQUEST['record_id'];
$mapping = $_REQUEST['mapping'];
$mapping = explode(',', $mapping);
//Error checking
//Load bean
$bean = loadBean($module);
$bean->retrieve($id);
require_once('include/connectors/ConnectorFactory.php');
$component = ConnectorFactory::getInstance($source_id);
//Create arguments
$args = array();
$field_defs = $bean->getFieldDefinitions();
foreach ($field_defs as $id=>$field) {
if (!empty($bean->$id)) {
$args[$id] = $bean->$id;
}
}
$beans = $component->fillBeans($args, $module);
if (!empty($beans) && !empty($mapping)) {
$results = array();
$count = 0;
foreach ($beans as $bean) {
foreach ($mapping as $field) {
$results[$count][$field] = $bean->$field;
}
$count++;
}
$json = getJSONobj();
echo $json->encode($results);
} else {
$GLOBALS['log']->error($GLOBALS['app_strings']['ERR_MISSING_MAPPING_ENTRY_FORM_MODULE']);
echo '';
}
}
public function action_SaveModifyProperties()
{
require_once('include/connectors/sources/SourceFactory.php');
$sources = array();
$properties = array();
foreach ($_REQUEST as $name=>$value) {
if (preg_match("/^source[0-9]+$/", $name, $matches)) {
$source_id = $value;
$properties = array();
foreach ($_REQUEST as $arg=>$val) {
if (preg_match("/^{$source_id}_(.*?)$/", $arg, $matches2)) {
$properties[$matches2[1]] = $val;
}
}
$source = SourceFactory::getSource($source_id);
if (!empty($properties)) {
$source->setProperties($properties);
$source->saveConfig();
}
}
}
require_once('include/connectors/utils/ConnectorUtils.php');
ConnectorUtils::updateMetaDataFiles();
// BEGIN SUGAR INT
if (empty($_REQUEST['from_unit_test'])) {
// END SUGAR INT
header("Location: index.php?action=ConnectorSettings&module=Connectors");
// BEGIN SUGAR INT
}
// END SUGAR INT
}
public function action_SaveModifyDisplay()
{
if (empty($_REQUEST['display_sources'])) {
return;
}
require_once('include/connectors/utils/ConnectorUtils.php');
require_once('include/connectors/sources/SourceFactory.php');
$connectors = ConnectorUtils::getConnectors();
$connector_keys = array_keys($connectors);
$modules_sources = ConnectorUtils::getDisplayConfig();
if (!is_array($modules_sources)) {
$modules_sources = (array)$modules_sources;
}
$sources = array();
$values = array();
$new_modules_sources = array();
if (!empty($_REQUEST['display_values'])) {
$display_values = explode(',', $_REQUEST['display_values']);
foreach ($display_values as $value) {
$entry = explode(':', $value);
$mod = get_module_from_singular($entry[1]); // get the internal module name
$new_modules_sources[$mod][$entry[0]] = $entry[0];
}
}
//These are the sources that were modified.
//We only update entries for these sources that have been changed
$display_sources = explode(',', $_REQUEST['display_sources']);
foreach ($display_sources as $source) {
$sources[$source] = $source;
} //foreach
$removedModules = array();
//Unset entries that have all sources removed
foreach ($modules_sources as $module => $source_entries) {
foreach ($source_entries as $source_id) {
if (!empty($sources[$source_id]) && empty($new_modules_sources[$module][$source_id])) {
unset($modules_sources[$module][$source_id]);
$removedModules[$module] = true;
}
}
}
$removedModules = array_keys($removedModules);
foreach ($removedModules as $key) {
if (empty($new_modules_sources[$key])) {
ConnectorUtils::cleanMetaDataFile($key);
}
}
//Update based on new_modules_sources
foreach ($new_modules_sources as $module => $enabled_sources) {
//If the module is not in $modules_sources add it there
if (empty($modules_sources[$module])) {
$modules_sources[$module] = $enabled_sources;
} else {
foreach ($enabled_sources as $source_id) {
if (empty($modules_sources[$module][$source_id])) {
$modules_sources[$module][$source_id] = $source_id;
}
} //foreach
}
} //foreach
//Should we just remove entries where all sources are disabled?
$unset_modules = array();
foreach ($modules_sources as $module => $mapping) {
if (empty($mapping)) {
$unset_modules[] = $module;
}
}
foreach ($unset_modules as $mod) {
unset($modules_sources[$mod]);
}
if (!write_array_to_file('modules_sources', $modules_sources, CONNECTOR_DISPLAY_CONFIG_FILE)) {
//Log error and return empty array
$GLOBALS['log']->fatal("Cannot write \$modules_sources to " . CONNECTOR_DISPLAY_CONFIG_FILE);
}
$sources_modules = array();
foreach ($modules_sources as $module => $source_entries) {
foreach ($source_entries as $id) {
$sources_modules[$id][$module] = $module;
}
}
//Now update the searchdefs and field mapping entries accordingly
require('modules/Connectors/metadata/searchdefs.php');
$originalSearchDefs = $searchdefs;
$connectorSearchDefs = ConnectorUtils::getSearchDefs();
$searchdefs = array();
foreach ($sources_modules as $source_id => $modules) {
foreach ($modules as $module) {
$searchdefs[$source_id][$module] = !empty($connectorSearchDefs[$source_id][$module]) ? $connectorSearchDefs[$source_id][$module] : (!empty($originalSearchDefs[$source_id][$module]) ? $originalSearchDefs[$source_id][$module] : array());
}
}
//Write the new searchdefs out
if (!write_array_to_file('searchdefs', $searchdefs, 'custom/modules/Connectors/metadata/searchdefs.php')) {
$GLOBALS['log']->fatal("Cannot write file custom/modules/Connectors/metadata/searchdefs.php");
}
//Unset the $_SESSION['searchDefs'] variable
if (isset($_SESSION['searchDefs'])) {
unset($_SESSION['searchDefs']);
}
//Clear mapping file if needed (this happens when all modules are removed from a source
foreach ($sources as $id) {
if (empty($sources_modules[$source])) {
//Now write the new mapping entry to the custom folder
$dir = $connectors[$id]['directory'];
if (!preg_match('/^custom\//', (string) $dir)) {
$dir = 'custom/' . $dir;
}
if (!file_exists((string)($dir))) {
mkdir_recursive((string)($dir));
}
$fakeMapping = array('beans' => array());
if (!write_array_to_file('mapping', $fakeMapping, "{$dir}/mapping.php")) {
$GLOBALS['log']->fatal("Cannot write file {$dir}/mapping.php");
}
$s = SourceFactory::getSource($id);
$s->saveMappingHook($fakeMapping);
} //if
} //foreach
//Now update the field mapping entries
foreach ($sources_modules as $id => $modules) {
$source = SourceFactory::getSource($id);
$mapping = $source->getMapping();
$mapped_modules = array_keys($mapping['beans']);
foreach ($mapped_modules as $module) {
if (empty($sources_modules[$id][$module])) {
unset($mapping['beans'][$module]);
}
}
//Remove modules from the mapping entries
foreach ($modules as $module) {
if (empty($mapping['beans'][$module])) {
$originalMapping = $source->getOriginalMapping();
if (empty($originalMapping['beans'][$module])) {
$defs = $source->getFieldDefs();
$keys = array_keys($defs);
$new_mapping_entry = array();
foreach ($keys as $key) {
$new_mapping_entry[$key] = '';
}
$mapping['beans'][$module] = $new_mapping_entry;
} else {
$mapping['beans'][$module] = $originalMapping['beans'][$module];
}
} //if
} //foreach
if ($id == 'ext_rest_twitter' || $id == 'ext_rest_facebook') {
$full_list = array_keys($mapping['beans']);
$new_modules = array_diff($full_list, $mapped_modules);
if (count($new_modules) > 0) {
foreach ($new_modules as $module) {
$field_name = substr($id, 9) . '_user_c';
$bean = BeanFactory::newBean($module);
if (!isset($bean->$field_name)) {
$this->add_social_field($module, $field_name);
}
}
}
unset($bean);
}
//Now write the new mapping entry to the custom folder
$dir = $connectors[$id]['directory'];
if (!preg_match('/^custom\//', (string) $dir)) {
$dir = 'custom/' . $dir;
}
if (!file_exists((string)($dir))) {
mkdir_recursive((string)($dir));
}
if (!write_array_to_file('mapping', $mapping, "{$dir}/mapping.php")) {
$GLOBALS['log']->fatal("Cannot write file {$dir}/mapping.php");
}
$source->saveMappingHook($mapping);
} //foreach
// save eapm configs
foreach ($connectors as $connector_name => $data) {
if (isset($sources[$connector_name]) && !empty($data["eapm"])) {
// if we touched it AND it has EAPM data
$connectors[$connector_name]["eapm"]["enabled"] = !empty($_REQUEST[$connector_name . "_external"]);
}
}
ConnectorUtils::saveConnectors($connectors);
ConnectorUtils::updateMetaDataFiles();
// BEGIN SUGAR INT
if (empty($_REQUEST['from_unit_test'])) {
// END SUGAR INT
header("Location: index.php?action=ConnectorSettings&module=Connectors");
// BEGIN SUGAR INT
}
// END SUGAR INT
}
/**
* action_SaveModifyMapping
*/
public function action_SaveModifyMapping()
{
$mapping_sources = !empty($_REQUEST['mapping_sources']) ? explode(',', $_REQUEST['mapping_sources']) : array();
$mapping_values = !empty($_REQUEST['mapping_values']) ? explode(',', $_REQUEST['mapping_values']) : array();
//Build the source->module->fields mapping
$source_modules_fields = array();
foreach ($mapping_values as $id) {
$parts = explode(':', $id);
$key_vals = explode('=', $parts[2]);
//Note the strtolwer call... we are lowercasing the key values
$source_modules_fields[$parts[0]][$parts[1]][strtolower($key_vals[0])] = $key_vals[1];
} //foreach
foreach ($mapping_sources as $source_id) {
if (empty($source_modules_fields[$source_id])) {
$source = SourceFactory::getSource($source_id);
$mapping = $source->getMapping();
foreach ($mapping['beans'] as $module=>$entry) {
$source_modules_fields[$source_id][$module] = array();
}
}
} //foreach
require_once('include/connectors/utils/ConnectorUtils.php');
$source_entries = ConnectorUtils::getConnectors();
require_once('include/connectors/sources/SourceFactory.php');
foreach ($source_modules_fields as $id=>$mapping_entry) {
//Insert the id mapping
foreach ($mapping_entry as $module=>$entry) {
$mapping_entry[$module]['id'] = 'id';
}
$source = SourceFactory::getSource($id);
$mapping = $source->getMapping();
$mapping['beans'] = $mapping_entry;
//Now write the new mapping entry to the custom folder
$dir = $source_entries[$id]['directory'];
if (!preg_match('/^custom\//', (string) $dir)) {
$dir = 'custom/' . $dir;
}
if (!file_exists((string)($dir))) {
mkdir_recursive((string)($dir));
}
if (!write_array_to_file('mapping', $mapping, "{$dir}/mapping.php")) {
$GLOBALS['log']->fatal("Cannot write file {$dir}/mapping.php");
}
$source->saveMappingHook($mapping);
}
//Rewrite the metadata files
ConnectorUtils::updateMetaDataFiles();
// BEGIN SUGAR INT
if (empty($_REQUEST['from_unit_test'])) {
// END SUGAR INT
header("Location: index.php?action=ConnectorSettings&module=Connectors");
// BEGIN SUGAR INT
}
// END SUGAR INT
}
public function action_RunTest()
{
$this->view = 'ajax';
$source_id = $_REQUEST['source_id'];
$source = SourceFactory::getSource($source_id);
$properties = array();
foreach ($_REQUEST as $name=>$value) {
if (preg_match("/^{$source_id}_(.*?)$/", $name, $matches)) {
$properties[$matches[1]] = $value;
}
}
$source->setProperties($properties);
$source->saveConfig();
//Call again and call init
$source = SourceFactory::getSource($source_id);
$source->init();
global $mod_strings;
try {
if ($source->isRequiredConfigFieldsForButtonSet() && $source->test()) {
echo $mod_strings['LBL_TEST_SOURCE_SUCCESS'];
} else {
echo $mod_strings['LBL_TEST_SOURCE_FAILED'];
}
} catch (Exception $ex) {
$GLOBALS['log']->fatal($ex->getMessage());
echo $ex->getMessage();
}
}
/**
* action_RetrieveSources
* Returns a JSON encoded format of the Connectors that are configured for the system
*
*/
public function action_RetrieveSources()
{
require_once('include/connectors/utils/ConnectorUtils.php');
$this->view = 'ajax';
$sources = ConnectorUtils:: getConnectors();
$results = array();
foreach ($sources as $id=>$entry) {
$results[$id] = !empty($entry['name']) ? $entry['name'] : $id;
}
$json = getJSONobj();
echo $json->encode($results);
}
public function add_social_field($module, $field_name)
{
$field = array(
array(
'name' => $field_name,
'label' => 'LBL_' . strtoupper($field_name),
'type' => 'varchar',
'module' => $module,
'ext1' => 'LIST',
'default_value' => '',
'mass_update' => false,
'required' => false,
'reportable' => false,
'audited' => false,
'importable' => 'false',
'duplicate_merge' => false,
)
);
$layout = [];
$layout[$module] = $field_name;
require_once('ModuleInstall/ModuleInstaller.php');
$moduleInstaller = new ModuleInstaller();
$moduleInstaller->install_custom_fields($field);
//$moduleInstaller->addFieldsToLayout($layout);
$this->create_panel_on_view('detailview', $field, $module, 'LBL_PANEL_SOCIAL_FEED');
/* now add it to the edit view. */
$this->create_panel_on_view('editview', $field, $module, 'LBL_PANEL_SOCIAL_FEED');
}
private function create_panel_on_view($view, $field, $module, $panel_name)
{
//require and create object.
require_once('modules/ModuleBuilder/parsers/ParserFactory.php');
$parser = ParserFactory::getParser($view, $module);
if (!array_key_exists($field['0']['name'], $parser->_fielddefs)) {
//add newly created fields to fielddefs as they wont be there.
$parser->_fielddefs[ $field['0']['name'] ] = $field['0'];
}
if (!array_key_exists($panel_name, $parser->_viewdefs['panels'])) {
//create the layout for the row.
$field_defs = array(0 => $field['0']['name']);
//add the row to the panel.
$panel = array( 0 => $field_defs );
//add the panel to the view.
$parser->_viewdefs['panels'][ $panel_name ] = $panel;
//save the panel.
} else {
//if the panel already exists we need to push items on to it.
foreach ($parser->_viewdefs['panels'][ $panel_name ] as $row_key => $row) {
foreach ($row as $key_field => $single_field) {
if ($single_field == "(empty)") {
$parser->_viewdefs['panels'][ $panel_name ][ $row_key ][ $key_field ] = $field['0']['name'];
}
}
}
}
$parser->handleSave(false);
}
}