type = $type; $this->filter = $filter; (empty($randomId) === true) ? $this->randomId = uniqid() : $this->randomId = $randomId; $this->refresh = $refresh; $this->width = $width; $this->height = $height; $this->search = $search; $this->group = $group; $this->dashboard = $dashboard; $this->hash = $hash; $this->publicUser = $publicUser; } /** * Run. * * @return void */ public function run() { ui_require_css_file('heatmap'); $settings = [ 'type' => 'GET', 'dataType' => 'html', 'url' => ui_get_full_url( 'ajax.php', false, false, false ), 'data' => [ 'page' => 'operation/heatmap', 'method' => 'showHeatmap', 'randomId' => $this->randomId, 'type' => $this->type, 'filter' => $this->filter, 'refresh' => $this->refresh, 'search' => $this->search, 'group' => $this->group, 'dashboard' => (int) $this->dashboard, 'auth_hash' => $this->hash, 'auth_class' => 'PandoraFMS\Dashboard\Manager', 'id_user' => $this->publicUser, ], ]; $style_dashboard = ''; if ($this->dashboard === true) { $style_dashboard = 'min-height: 0px'; } echo '
'; ?> '; } /** * Setter for filter * * @param array $filter Filter. * * @return void */ public function setFilter(array $filter) { $this->filter = $filter; } /** * Setter for type * * @param integer $type Type. * * @return void */ public function setType(int $type) { $this->type = $type; } /** * Setter for refresh * * @param integer $refresh Refresh. * * @return void */ public function setRefresh(int $refresh) { $this->refresh = $refresh; } /** * Getter for randomId * * @return string */ public function getRandomId() { return $this->randomId; } /** * Get all agents * * @return array */ protected function getAllAgents() { global $config; $filter['disabled'] = 0; $alias = ''; if (empty($this->search) === false) { $alias = ' AND alias LIKE "%'.$this->search.'%"'; } $id_user_groups = ''; if (users_is_admin() === false) { $user_groups = array_keys(users_get_groups($config['user'], 'AR', false)); if (empty($user_groups) === false) { $id_user_groups = ' AND id_grupo IN ('.implode(',', $user_groups).')'; } } $id_grupo = ''; if (empty($this->filter) === false && empty(current($this->filter)) === false) { $id_grupo = ' AND id_grupo IN ('.implode(',', $this->filter).')'; } // All agents. $sql = sprintf( 'SELECT DISTINCT id_agente as id,alias,id_grupo,normal_count,warning_count,critical_count, unknown_count,notinit_count,total_count,fired_count, (SELECT last_status_change FROM tagente_estado WHERE id_agente = tagente.id_agente ORDER BY last_status_change DESC LIMIT 1) AS last_status_change FROM tagente WHERE `disabled` = 0 %s %s %s ORDER BY id_grupo,id_agente ASC', $alias, $id_user_groups, $id_grupo ); $agents = []; if (is_metaconsole() === true) { $nodes = metaconsole_get_connections(); $cont = 0; foreach ($nodes as $node) { try { $nd = new Node($node['id']); $nd->connect(); $result = db_get_all_rows_sql($sql); // Agent status. foreach ($result as $agent) { if ($agent['total_count'] === 0 || $agent['total_count'] === $agent['notinit_count']) { $status = 'notinit'; } else if ($agent['critical_count'] > 0) { $status = 'critical'; } else if ($agent['warning_count'] > 0) { $status = 'warning'; } else if ($agent['unknown_count'] > 0) { $status = 'unknown'; } else { $status = 'normal'; } if ($agent['last_status_change'] != 0) { $seconds = (time() - $agent['last_status_change']); if ($seconds >= SECONDS_1DAY) { $status .= '_10'; } else if ($seconds >= 77760) { $status .= '_9'; } else if ($seconds >= 69120) { $status .= '_8'; } else if ($seconds >= 60480) { $status .= '_7'; } else if ($seconds >= 51840) { $status .= '_6'; } else if ($seconds >= 43200) { $status .= '_5'; } else if ($seconds >= 34560) { $status .= '_4'; } else if ($seconds >= 25920) { $status .= '_3'; } else if ($seconds >= 17280) { $status .= '_2'; } else if ($seconds >= 8640) { $status .= '_1'; } } $agents[$cont] = $agent; $agents[$cont]['status'] = $status; $agents[$cont]['server'] = $node['id']; ++$cont; } } catch (\Exception $e) { $nd->disconnect(); $agents = []; } finally { $nd->disconnect(); } } } else { $result = db_get_all_rows_sql($sql); // Agent status. foreach ($result as $key => $agent) { if ($agent['total_count'] === 0 || $agent['total_count'] === $agent['notinit_count']) { $status = 'notinit'; } else if ($agent['critical_count'] > 0) { $status = 'critical'; } else if ($agent['warning_count'] > 0) { $status = 'warning'; } else if ($agent['unknown_count'] > 0) { $status = 'unknown'; } else { $status = 'normal'; } if ($agent['last_status_change'] != 0) { $seconds = (time() - $agent['last_status_change']); if ($seconds >= SECONDS_1DAY) { $status .= '_10'; } else if ($seconds >= 77760) { $status .= '_9'; } else if ($seconds >= 69120) { $status .= '_8'; } else if ($seconds >= 60480) { $status .= '_7'; } else if ($seconds >= 51840) { $status .= '_6'; } else if ($seconds >= 43200) { $status .= '_5'; } else if ($seconds >= 34560) { $status .= '_4'; } else if ($seconds >= 25920) { $status .= '_3'; } else if ($seconds >= 17280) { $status .= '_2'; } else if ($seconds >= 8640) { $status .= '_1'; } } $agents[$key] = $agent; $agents[$key]['status'] = $status; } } return $agents; } /** * Get all modules * * @return array */ protected function getAllModulesByGroup() { global $config; $filter_group = ''; if (empty($this->filter) === false && current($this->filter) != -1) { $filter_group = 'AND am.id_module_group IN ('.implode(',', $this->filter).')'; } $filter_name = ''; if (empty($this->search) === false) { $filter_name = 'AND nombre LIKE "%'.$this->search.'%"'; } $id_user_groups = ''; if (users_is_admin() === false) { $user_groups = array_keys(users_get_groups($config['user'], 'AR', false)); if (empty($user_groups) === false) { $id_user_groups = sprintf( 'INNER JOIN tagente a ON a.id_agente = ae.id_agente AND a.id_grupo IN (%s)', implode(',', $user_groups) ); } } // All modules. $sql = sprintf( 'SELECT am.id_agente_modulo AS id, ae.estado AS `status`, am.id_module_group AS id_grupo, ae.last_status_change FROM tagente_modulo am INNER JOIN tagente_estado ae ON am.id_agente_modulo = ae.id_agente_modulo %s WHERE am.disabled = 0 %s %s GROUP BY am.id_module_group, am.id_agente_modulo', $id_user_groups, $filter_group, $filter_name ); if (is_metaconsole() === true) { $nodes = metaconsole_get_connections(); $cont = 0; $result = []; foreach ($nodes as $node) { try { $nd = new Node($node['id']); $nd->connect(); $modules = db_get_all_rows_sql($sql); // Module status. foreach ($modules as $key => $module) { $status = ''; switch ($module['status']) { case AGENT_MODULE_STATUS_CRITICAL_BAD: case AGENT_MODULE_STATUS_CRITICAL_ALERT: case 1: case 100: $status = 'critical'; break; case AGENT_MODULE_STATUS_NORMAL: case AGENT_MODULE_STATUS_NORMAL_ALERT: case 0: case 300: $status = 'normal'; break; case AGENT_MODULE_STATUS_WARNING: case AGENT_MODULE_STATUS_WARNING_ALERT: case 2: case 200: $status = 'warning'; break; default: case AGENT_MODULE_STATUS_UNKNOWN: case 3: $status = 'unknown'; break; case AGENT_MODULE_STATUS_NOT_INIT: case 5: $status = 'notinit'; break; } if ($module['last_status_change'] != 0) { $seconds = (time() - $module['last_status_change']); if ($seconds >= SECONDS_1DAY) { $status .= '_10'; } else if ($seconds >= 77760) { $status .= '_9'; } else if ($seconds >= 69120) { $status .= '_8'; } else if ($seconds >= 60480) { $status .= '_7'; } else if ($seconds >= 51840) { $status .= '_6'; } else if ($seconds >= 43200) { $status .= '_5'; } else if ($seconds >= 34560) { $status .= '_4'; } else if ($seconds >= 25920) { $status .= '_3'; } else if ($seconds >= 17280) { $status .= '_2'; } else if ($seconds >= 8640) { $status .= '_1'; } } $result[$cont] = $module; $result[$cont]['status'] = $status; $result[$cont]['server'] = $node['id']; ++$cont; } } catch (\Exception $e) { $nd->disconnect(); } finally { $nd->disconnect(); } } } else { $result = db_get_all_rows_sql($sql); // Module status. foreach ($result as $key => $module) { $status = ''; switch ($module['status']) { case AGENT_MODULE_STATUS_CRITICAL_BAD: case AGENT_MODULE_STATUS_CRITICAL_ALERT: case 1: case 100: $status = 'critical'; break; case AGENT_MODULE_STATUS_NORMAL: case AGENT_MODULE_STATUS_NORMAL_ALERT: case 0: case 300: $status = 'normal'; break; case AGENT_MODULE_STATUS_WARNING: case AGENT_MODULE_STATUS_WARNING_ALERT: case 2: case 200: $status = 'warning'; break; default: case AGENT_MODULE_STATUS_UNKNOWN: case 3: $status = 'unknown'; break; case AGENT_MODULE_STATUS_NOT_INIT: case 5: $status = 'notinit'; break; } if ($module['last_status_change'] != 0) { $seconds = (time() - $module['last_status_change']); if ($seconds >= SECONDS_1DAY) { $status .= '_10'; } else if ($seconds >= 77760) { $status .= '_9'; } else if ($seconds >= 69120) { $status .= '_8'; } else if ($seconds >= 60480) { $status .= '_7'; } else if ($seconds >= 51840) { $status .= '_6'; } else if ($seconds >= 43200) { $status .= '_5'; } else if ($seconds >= 34560) { $status .= '_4'; } else if ($seconds >= 25920) { $status .= '_3'; } else if ($seconds >= 17280) { $status .= '_2'; } else if ($seconds >= 8640) { $status .= '_1'; } } $result[$key]['status'] = $status; } } return $result; } /** * Get all modules * * @return array */ protected function getAllModulesByTag() { global $config; $filter_tag = ''; if (empty($this->filter) === false && $this->filter[0] !== '0') { $tags = implode(',', $this->filter); $filter_tag .= ' AND tm.id_tag IN ('.$tags.')'; } $filter_name = ''; if (empty($this->search) === false) { $filter_name = 'AND nombre LIKE "%'.$this->search.'%"'; } $id_user_groups = ''; if (users_is_admin() === false) { $user_groups = array_keys(users_get_groups($config['user'], 'AR', false)); if (empty($user_groups) === false) { $id_user_groups = sprintf( 'INNER JOIN tagente a ON a.id_agente = ae.id_agente AND a.id_grupo IN (%s)', implode(',', $user_groups) ); } } // All modules. $sql = sprintf( 'SELECT ae.id_agente_modulo AS id, ae.estado AS `status`, tm.id_tag AS id_grupo, ae.last_status_change FROM tagente_estado ae %s INNER JOIN ttag_module tm ON tm.id_agente_modulo = ae.id_agente_modulo WHERE 1=1 %s %s GROUP BY ae.id_agente_modulo', $id_user_groups, $filter_tag, $filter_name ); if (is_metaconsole() === true) { $nodes = metaconsole_get_connections(); $result = []; $cont = 0; foreach ($nodes as $node) { try { $nd = new Node($node['id']); $nd->connect(); $modules = db_get_all_rows_sql($sql); // Module status. foreach ($modules as $key => $module) { $status = ''; switch ($module['status']) { case AGENT_MODULE_STATUS_CRITICAL_BAD: case AGENT_MODULE_STATUS_CRITICAL_ALERT: case 1: case 100: $status = 'critical'; break; case AGENT_MODULE_STATUS_NORMAL: case AGENT_MODULE_STATUS_NORMAL_ALERT: case 0: case 300: $status = 'normal'; break; case AGENT_MODULE_STATUS_WARNING: case AGENT_MODULE_STATUS_WARNING_ALERT: case 2: case 200: $status = 'warning'; break; default: case AGENT_MODULE_STATUS_UNKNOWN: case 3: $status = 'unknown'; break; case AGENT_MODULE_STATUS_NOT_INIT: case 5: $status = 'notinit'; break; } if ($module['last_status_change'] != 0) { $seconds = (time() - $module['last_status_change']); if ($seconds >= SECONDS_1DAY) { $status .= '_10'; } else if ($seconds >= 77760) { $status .= '_9'; } else if ($seconds >= 69120) { $status .= '_8'; } else if ($seconds >= 60480) { $status .= '_7'; } else if ($seconds >= 51840) { $status .= '_6'; } else if ($seconds >= 43200) { $status .= '_5'; } else if ($seconds >= 34560) { $status .= '_4'; } else if ($seconds >= 25920) { $status .= '_3'; } else if ($seconds >= 17280) { $status .= '_2'; } else if ($seconds >= 8640) { $status .= '_1'; } } $result[$cont] = $module; $result[$cont]['status'] = $status; $result[$cont]['server'] = $node['id']; ++$cont; } } catch (\Exception $e) { $nd->disconnect(); } finally { $nd->disconnect(); } } } else { $result = db_get_all_rows_sql($sql); // Module status. foreach ($result as $key => $module) { $status = ''; switch ($module['status']) { case AGENT_MODULE_STATUS_CRITICAL_BAD: case AGENT_MODULE_STATUS_CRITICAL_ALERT: case 1: case 100: $status = 'critical'; break; case AGENT_MODULE_STATUS_NORMAL: case AGENT_MODULE_STATUS_NORMAL_ALERT: case 0: case 300: $status = 'normal'; break; case AGENT_MODULE_STATUS_WARNING: case AGENT_MODULE_STATUS_WARNING_ALERT: case 2: case 200: $status = 'warning'; break; default: case AGENT_MODULE_STATUS_UNKNOWN: case 3: $status = 'unknown'; break; case AGENT_MODULE_STATUS_NOT_INIT: case 5: $status = 'notinit'; break; } if ($module['last_status_change'] != 0) { $seconds = (time() - $module['last_status_change']); if ($seconds >= SECONDS_1DAY) { $status .= '_10'; } else if ($seconds >= 77760) { $status .= '_9'; } else if ($seconds >= 69120) { $status .= '_8'; } else if ($seconds >= 60480) { $status .= '_7'; } else if ($seconds >= 51840) { $status .= '_6'; } else if ($seconds >= 43200) { $status .= '_5'; } else if ($seconds >= 34560) { $status .= '_4'; } else if ($seconds >= 25920) { $status .= '_3'; } else if ($seconds >= 17280) { $status .= '_2'; } else if ($seconds >= 8640) { $status .= '_1'; } } $result[$key]['status'] = $status; } } return $result; } /** * Get all modules group by agents * * @return array */ protected function getAllModulesByAgents() { global $config; $filter_name = ''; if (empty($this->search) === false) { $filter_name = 'AND nombre LIKE "%'.$this->search.'%"'; } $id_user_groups = ''; if (users_is_admin() === false) { $user_groups = array_keys(users_get_groups($config['user'], 'AR', false)); if (empty($user_groups) === false) { if (empty($this->filter) === false && empty(current($this->filter)) === false) { $user_groups = array_intersect($this->filter, $user_groups); $id_user_groups = sprintf( 'INNER JOIN tagente a ON a.id_agente = ae.id_agente AND a.id_grupo IN (%s)', implode(',', $user_groups) ); } else { $id_user_groups = sprintf( 'INNER JOIN tagente a ON a.id_agente = ae.id_agente AND a.id_grupo IN (%s)', implode(',', $user_groups) ); } } } else { if (empty($this->filter) === false && empty(current($this->filter)) === false) { $id_user_groups = sprintf( 'INNER JOIN tagente a ON a.id_agente = ae.id_agente AND a.id_grupo IN (%s)', implode(',', $this->filter) ); } } // All modules. $sql = sprintf( 'SELECT am.id_agente_modulo AS id, ae.estado AS `status`, am.id_agente AS id_grupo, ae.last_status_change FROM tagente_modulo am INNER JOIN tagente_estado ae ON am.id_agente_modulo = ae.id_agente_modulo %s WHERE am.disabled = 0 %s GROUP BY ae.id_agente_modulo ORDER BY id_grupo', $id_user_groups, $filter_name ); if (is_metaconsole() === true) { $result = []; $nodes = metaconsole_get_connections(); $cont = 0; foreach ($nodes as $node) { try { $nd = new Node($node['id']); $nd->connect(); $modules = db_get_all_rows_sql($sql); // Module status. foreach ($modules as $key => $module) { $status = ''; switch ($module['status']) { case AGENT_MODULE_STATUS_CRITICAL_BAD: case AGENT_MODULE_STATUS_CRITICAL_ALERT: case 1: case 100: $status = 'critical'; break; case AGENT_MODULE_STATUS_NORMAL: case AGENT_MODULE_STATUS_NORMAL_ALERT: case 0: case 300: $status = 'normal'; break; case AGENT_MODULE_STATUS_WARNING: case AGENT_MODULE_STATUS_WARNING_ALERT: case 2: case 200: $status = 'warning'; break; default: case AGENT_MODULE_STATUS_UNKNOWN: case 3: $status = 'unknown'; break; case AGENT_MODULE_STATUS_NOT_INIT: case 5: $status = 'notinit'; break; } if ($module['last_status_change'] != 0) { $seconds = (time() - $module['last_status_change']); if ($seconds >= SECONDS_1DAY) { $status .= '_10'; } else if ($seconds >= 77760) { $status .= '_9'; } else if ($seconds >= 69120) { $status .= '_8'; } else if ($seconds >= 60480) { $status .= '_7'; } else if ($seconds >= 51840) { $status .= '_6'; } else if ($seconds >= 43200) { $status .= '_5'; } else if ($seconds >= 34560) { $status .= '_4'; } else if ($seconds >= 25920) { $status .= '_3'; } else if ($seconds >= 17280) { $status .= '_2'; } else if ($seconds >= 8640) { $status .= '_1'; } } $result[$cont] = $module; $result[$cont]['status'] = $status; $result[$cont]['server'] = $node['id']; ++$cont; } } catch (\Exception $e) { $nd->disconnect(); $result = []; } finally { $nd->disconnect(); } } } else { $result = db_get_all_rows_sql($sql); // Module status. foreach ($result as $key => $module) { $status = ''; switch ($module['status']) { case AGENT_MODULE_STATUS_CRITICAL_BAD: case AGENT_MODULE_STATUS_CRITICAL_ALERT: case 1: case 100: $status = 'critical'; break; case AGENT_MODULE_STATUS_NORMAL: case AGENT_MODULE_STATUS_NORMAL_ALERT: case 0: case 300: $status = 'normal'; break; case AGENT_MODULE_STATUS_WARNING: case AGENT_MODULE_STATUS_WARNING_ALERT: case 2: case 200: $status = 'warning'; break; default: case AGENT_MODULE_STATUS_UNKNOWN: case 3: $status = 'unknown'; break; case AGENT_MODULE_STATUS_NO_DATA: case AGENT_MODULE_STATUS_NOT_INIT: case 5: $status = 'notinit'; break; } if ($module['last_status_change'] != 0) { $seconds = (time() - $module['last_status_change']); if ($seconds >= SECONDS_1DAY) { $status .= '_10'; } else if ($seconds >= 77760) { $status .= '_9'; } else if ($seconds >= 69120) { $status .= '_8'; } else if ($seconds >= 60480) { $status .= '_7'; } else if ($seconds >= 51840) { $status .= '_6'; } else if ($seconds >= 43200) { $status .= '_5'; } else if ($seconds >= 34560) { $status .= '_4'; } else if ($seconds >= 25920) { $status .= '_3'; } else if ($seconds >= 17280) { $status .= '_2'; } else if ($seconds >= 8640) { $status .= '_1'; } } $result[$key]['status'] = $status; } } return $result; } /** * GetData * * @return array */ public function getData() { switch ($this->type) { case 3: $data = $this->getAllModulesByAgents(); break; case 2: $data = $this->getAllModulesByGroup(); break; case 1: $data = $this->getAllModulesByTag(); break; case 0: default: $data = $this->getAllAgents(); break; } return $data; } /** * GetDataJson * * @return json */ public function getDataJson() { $return = $this->getData(); echo json_encode($return); return ''; } /** * Get class by status * * @param integer $status Status. * * @return string */ protected function statusColour(int $status) { switch ($status) { case AGENT_STATUS_CRITICAL: $return = 'critical'; break; case AGENT_STATUS_WARNING: $return = 'warning'; break; case AGENT_STATUS_UNKNOWN: $return = 'unknown'; break; case AGENT_STATUS_NOT_INIT: $return = 'notinit'; break; case AGENT_STATUS_NORMAL: default: $return = 'normal'; break; } return $return; } /** * Get max. number of y-axis * * @param integer $total Total. * @param float $relation Aspect relation. * * @return integer */ protected function getYAxis(int $total, float $relation) { $yAxis = sqrt(($total / $relation)); return $yAxis; } /** * Checks if target method is available to be called using AJAX. * * @param string $method Target method. * * @return boolean True allowed, false not. */ public function ajaxMethod(string $method): bool { return in_array($method, $this->AJAXMethods); } /** * ShowHeatmap * * @return void */ public function showHeatmap() { $result = $this->getData(); if (empty($result) === true) { echo '
'.__('No data found').'
'; return; } $count_result = count($result); $scale = ($this->width / $this->height); $Yaxis = $this->getYAxis($count_result, $scale); $Yaxis_font_scale = $Yaxis; if ($count_result <= 3) { $Xaxis = $count_result; $Yaxis = 1; } else { $Xaxis = (int) ceil($Yaxis * $scale); $Yaxis = ceil($Yaxis); } $viewBox = sprintf( '0 0 %d %d', $Xaxis, $Yaxis ); echo ''; $groups = []; $contX = 0; $contY = 0; $cont = 1; foreach ($result as $value) { $name = $value['id']; if (empty($value['server']) === false) { $name .= '|'.$value['server']; } echo ''; $contX++; if ($contX >= $Xaxis) { $contY++; $contX = 0; } if (empty($groups[$value['id_grupo']]) === true) { $groups[$value['id_grupo']] = 1; } else { $groups[$value['id_grupo']] += 1; } $cont++; } ?> 1 && $this->group === 1 && $this->dashboard === false) { $x_back = 0; $y_back = 0; $x_text_correction = 0.08; if ($count_result <= 10) { $fontSize = 0.2; $stroke = 'small-stroke'; } else if ($count_result > 10 && $count_result <= 100) { $fontSize = 0.18; $stroke = 'tiny-stroke'; } else if ($count_result > 100 && $count_result <= 1000) { $fontSize = 0.16; $stroke = 'medium-stroke'; } else if ($count_result > 1000 && $count_result <= 10000) { $fontSize = 0.14; $stroke = 'big-stroke'; } else { $fontSize = 0.12; $stroke = 'huge-stroke'; } echo ''; foreach ($groups as $key => $group) { $name = ''; switch ($this->type) { case 3: $name = agents_get_alias($key); break; case 2: $name = modules_get_modulegroup_name($key); break; case 1: $name = tags_get_name($key); break; case 0: default: $name = groups_get_name($key); break; } if (($x_back + $group) <= $Xaxis) { $x_position = ($x_back + $group); $y_position = $y_back; if ($y_back === 0 && $x_back === 0) { $points = sprintf( '%d,%d %d,%d', $x_back, $y_back, $x_back, ($y_back + 1) ); echo ''; } $points = sprintf( '%d,%d %d,%d %d,%d', $x_back, ($y_position + 1), $x_position, ($y_position + 1), $x_position, $y_back ); echo ''; // Name. echo ''.$name.''; $x_back = $x_position; if ($x_position === $Xaxis) { $points = sprintf( '%d,%d %d,%d', $x_position, $y_back, $x_position, ($y_back + 1) ); echo ''; $y_back++; $x_back = 0; } } else { $round = (int) floor(($x_back + $group) / $Xaxis); $y_position = ($round + $y_back); if ($round === 1) { // One line. $x_position = (($x_back + $group) - $Xaxis); if ($x_position <= $x_back) { // Bottom line. $points = sprintf( '%d,%d %d,%d', $x_back, $y_position, $Xaxis, ($y_position) ); echo ''; } // Bottom of last line. $points = sprintf( '%d,%d %d,%d', 0, ($y_position + 1), $x_position, ($y_position + 1) ); echo ''; // Name. echo ''.$name.''; // Bottom-right of last line. $points = sprintf( '%d,%d %d,%d', $x_position, ($y_position), $x_position, ($y_position + 1) ); echo ''; if ($x_position > $x_back) { // Bottom-top of last line. $points = sprintf( '%d,%d %d,%d', $x_position, ($y_position), $Xaxis, ($y_position) ); echo ''; } } else { // Two or more lines. $x_position = (($x_back + $group) - ($Xaxis * $round)); if ($x_position === 0) { $x_position = $Xaxis; } // Bottom of last line. $points = sprintf( '%d,%d %d,%d', 0, ($y_position + 1), $x_position, ($y_position + 1) ); echo ''; // Bottom-right of last line. $points = sprintf( '%d,%d %d,%d', $x_position, ($y_position), $x_position, ($y_position + 1) ); echo ''; // Name. echo ''.$name.''; // Bottom-top of last line. $points = sprintf( '%d,%d %d,%d', $x_position, ($y_position), $Xaxis, ($y_position) ); echo ''; } if ($x_position === $Xaxis) { $x_position = 0; } $x_back = $x_position; $y_back = $y_position; } } } echo ''; // Dialog. echo '
'; } }