'General_AverageOrderValue', 'nb_conversions' => 'Goals_ColumnConversions', 'conversion_rate' => 'General_ColumnConversionRate', 'revenue' => 'General_TotalRevenue', 'items' => 'General_PurchasedProducts', ); /** * @var Translator */ private $translator; /** * @var TranslationHelper */ private $translationHelper; private function formatConversionRate($conversionRate, $columnName = 'conversion_rate') { if ($conversionRate instanceof DataTable) { if ($conversionRate->getRowsCount() == 0) { $conversionRate = 0; } else { $conversionRate = $conversionRate->getFirstRow()->getColumn($columnName); } } if (!is_numeric($conversionRate)) { $conversionRate = sprintf('%.' . self::CONVERSION_RATE_PRECISION . 'f%%', $conversionRate); } return $conversionRate; } public function __construct(Translator $translator, TranslationHelper $translationHelper) { parent::__construct(); $this->translator = $translator; $this->translationHelper = $translationHelper; $this->idSite = Common::getRequestVar('idSite', null, 'int'); $this->goals = API::getInstance()->getGoals($this->idSite); } public function widgetGoalReport() { $view = $this->getGoalReportView($idGoal = Common::getRequestVar('idGoal', null, 'string')); $view->displayFullReport = false; return $view->render(); } public function goalReport() { $view = $this->getGoalReportView($idGoal = Common::getRequestVar('idGoal', null, 'string')); $view->displayFullReport = true; return $view->render(); } protected function getGoalReportView($idGoal = false) { $view = new View('@Goals/getGoalReportView'); if ($idGoal == Piwik::LABEL_ID_GOAL_IS_ECOMMERCE_ORDER) { $goalDefinition['name'] = $this->translator->translate('Goals_Ecommerce'); $goalDefinition['allow_multiple'] = true; $ecommerce = $view->ecommerce = true; } else { if (!isset($this->goals[$idGoal])) { Piwik::redirectToModule('Goals', 'index', array('idGoal' => null)); } $goalDefinition = $this->goals[$idGoal]; } $this->setGeneralVariablesView($view); $goal = $this->getMetricsForGoal($idGoal); foreach ($goal as $name => $value) { $view->$name = $value; } if ($idGoal == Piwik::LABEL_ID_GOAL_IS_ECOMMERCE_ORDER) { $goal = $this->getMetricsForGoal(Piwik::LABEL_ID_GOAL_IS_ECOMMERCE_CART); foreach ($goal as $name => $value) { $name = 'cart_' . $name; $view->$name = $value; } } $view->showHeadline = false; $view->idGoal = $idGoal; $view->goalName = $goalDefinition['name']; $view->goalAllowMultipleConversionsPerVisit = $goalDefinition['allow_multiple']; $view->graphEvolution = $this->getEvolutionGraph(array(), $idGoal, array('nb_conversions')); $view->nameGraphEvolution = 'Goals.getEvolutionGraph' . $idGoal; $view->topDimensions = $this->getTopDimensions($idGoal); $goalMetrics = Request::processRequest('Goals.get', array('idGoal' => $idGoal)); // conversion rate for new and returning visitors $view->conversion_rate_returning = $this->formatConversionRate($goalMetrics, 'conversion_rate_returning_visit'); $view->conversion_rate_new = $this->formatConversionRate($goalMetrics, 'conversion_rate_new_visit'); $view->goalReportsByDimension = $this->getGoalReportsByDimensionTable( $view->nb_conversions, isset($ecommerce), !empty($view->cart_nb_conversions)); return $view; } public function index() { $view = $this->getOverviewView(); // unsanitize goal names and other text data (not done in API so as not to break // any other code/cause security issues) $goals = $this->goals; $view->goalsJSON = json_encode($goals); $view->ecommerceEnabled = $this->site->isEcommerceEnabled(); $view->displayFullReport = true; return $view->render(); } public function manage() { Piwik::checkUserHasAdminAccess($this->idSite); $view = new View('@Goals/manageGoals'); $this->setGeneralVariablesView($view); $this->setEditGoalsViewVariables($view); $this->setUserCanEditGoals($view); return $view->render(); } public function widgetGoalsOverview() { $view = $this->getOverviewView(); $view->displayFullReport = false; return $view->render(); } protected function getOverviewView() { $view = new View('@Goals/getOverviewView'); $this->setGeneralVariablesView($view); $view->graphEvolution = $this->getEvolutionGraph(array(), false, array('nb_conversions')); $view->nameGraphEvolution = 'GoalsgetEvolutionGraph'; // sparkline for the historical data of the above values $view->urlSparklineConversions = $this->getUrlSparkline('getEvolutionGraph', array('columns' => array('nb_conversions'), 'idGoal' => '')); $view->urlSparklineConversionRate = $this->getUrlSparkline('getEvolutionGraph', array('columns' => array('conversion_rate'), 'idGoal' => '')); $view->urlSparklineRevenue = $this->getUrlSparkline('getEvolutionGraph', array('columns' => array('revenue'), 'idGoal' => '')); // Pass empty idGoal will return Goal overview $request = new Request("method=Goals.get&format=original&idGoal="); $datatable = $request->process(); $dataRow = $datatable->getFirstRow(); $view->nb_conversions = $dataRow->getColumn('nb_conversions'); $view->nb_visits_converted = $dataRow->getColumn('nb_visits_converted'); $view->conversion_rate = $this->formatConversionRate($dataRow->getColumn('conversion_rate')); $view->revenue = $dataRow->getColumn('revenue'); $goalMetrics = array(); foreach ($this->goals as $idGoal => $goal) { $goalMetrics[$idGoal] = $this->getMetricsForGoal($idGoal); $goalMetrics[$idGoal]['name'] = $goal['name']; $goalMetrics[$idGoal]['goalAllowMultipleConversionsPerVisit'] = $goal['allow_multiple']; } $view->goalMetrics = $goalMetrics; $view->goals = $this->goals; $view->goalReportsByDimension = $this->getGoalReportsByDimensionTable( $view->nb_conversions, $ecommerce = false, !empty($view->cart_nb_conversions)); return $view; } public function getLastNbConversionsGraph() { $view = $this->getLastUnitGraph($this->pluginName, __FUNCTION__, 'Goals.getConversions'); return $this->renderView($view); } public function getLastConversionRateGraph() { $view = $this->getLastUnitGraph($this->pluginName, __FUNCTION__, 'Goals.getConversionRate'); return $this->renderView($view); } public function getLastRevenueGraph() { $view = $this->getLastUnitGraph($this->pluginName, __FUNCTION__, 'Goals.getRevenue'); return $this->renderView($view); } public function addNewGoal() { $view = new View('@Goals/addNewGoal'); $this->setGeneralVariablesView($view); $this->setUserCanEditGoals($view); $view->onlyShowAddNewGoal = true; return $view->render(); } public function editGoals() { $view = new View('@Goals/editGoals'); $this->setGeneralVariablesView($view); $this->setEditGoalsViewVariables($view); $this->setUserCanEditGoals($view); return $view->render(); } public function getEvolutionGraph(array $columns = array(), $idGoal = false, array $defaultColumns = array()) { if (empty($columns)) { $columns = Common::getRequestVar('columns', false); if (false !== $columns) { $columns = Piwik::getArrayFromApiParameter($columns); } } if (false !== $columns) { $columns = !is_array($columns) ? array($columns) : $columns; } if (empty($idGoal)) { $idGoal = Common::getRequestVar('idGoal', false, 'string'); } $view = $this->getLastUnitGraph($this->pluginName, __FUNCTION__, 'Goals.get'); $view->requestConfig->request_parameters_to_modify['idGoal'] = $idGoal; $nameToLabel = $this->goalColumnNameToLabel; if ($idGoal == Piwik::LABEL_ID_GOAL_IS_ECOMMERCE_ORDER) { $nameToLabel['nb_conversions'] = 'General_EcommerceOrders'; } elseif ($idGoal == Piwik::LABEL_ID_GOAL_IS_ECOMMERCE_CART) { $nameToLabel['nb_conversions'] = $this->translator->translate('General_VisitsWith', $this->translator->translate('Goals_AbandonedCart')); $nameToLabel['conversion_rate'] = $nameToLabel['nb_conversions']; $nameToLabel['revenue'] = $this->translator->translate('Goals_LeftInCart', $this->translator->translate('General_ColumnRevenue')); $nameToLabel['items'] = $this->translator->translate('Goals_LeftInCart', $this->translator->translate('Goals_Products')); } $selectableColumns = array('nb_conversions', 'conversion_rate', 'revenue'); if ($this->site->isEcommerceEnabled()) { $selectableColumns[] = 'items'; $selectableColumns[] = 'avg_order_revenue'; } foreach (array_merge($columns ? $columns : array(), $selectableColumns) as $columnName) { $columnTranslation = ''; // find the right translation for this column, eg. find 'revenue' if column is Goal_1_revenue foreach ($nameToLabel as $metric => $metricTranslation) { if (strpos($columnName, $metric) !== false) { $columnTranslation = $this->translator->translate($metricTranslation); break; } } if (!empty($idGoal) && isset($this->goals[$idGoal])) { $goalName = $this->goals[$idGoal]['name']; $columnTranslation = "$columnTranslation (" . $this->translator->translate('Goals_GoalX', "$goalName") . ")"; } $view->config->translations[$columnName] = $columnTranslation; } if (!empty($columns)) { $view->config->columns_to_display = $columns; } elseif (empty($view->config->columns_to_display) && !empty($defaultColumns)) { $view->config->columns_to_display = $defaultColumns; } $view->config->selectable_columns = $selectableColumns; $langString = $idGoal ? 'Goals_SingleGoalOverviewDocumentation' : 'Goals_GoalsOverviewDocumentation'; $view->config->documentation = $this->translator->translate($langString, '
'); return $this->renderView($view); } protected function getTopDimensions($idGoal) { $columnNbConversions = 'goal_' . $idGoal . '_nb_conversions'; $columnConversionRate = 'goal_' . $idGoal . '_conversion_rate'; $topDimensionsToLoad = array(); if (\Piwik\Plugin\Manager::getInstance()->isPluginActivated('UserCountry')) { $topDimensionsToLoad += array( 'country' => 'UserCountry.getCountry', ); } $keywordNotDefinedString = ''; if (\Piwik\Plugin\Manager::getInstance()->isPluginActivated('Referrers')) { $keywordNotDefinedString = APIReferrers::getKeywordNotDefinedString(); $topDimensionsToLoad += array( 'keyword' => 'Referrers.getKeywords', 'website' => 'Referrers.getWebsites', ); } $topDimensions = array(); foreach ($topDimensionsToLoad as $dimensionName => $apiMethod) { $request = new Request("method=$apiMethod &format=original &filter_update_columns_when_show_all_goals=1 &idGoal=" . AddColumnsProcessedMetricsGoal::GOALS_FULL_TABLE . " &filter_sort_order=desc &filter_sort_column=$columnNbConversions" . // select a couple more in case some are not valid (ie. conversions==0 or they are "Keyword not defined") "&filter_limit=" . (self::COUNT_TOP_ROWS_TO_DISPLAY + 2)); $datatable = $request->process(); $topDimension = array(); $count = 0; foreach ($datatable->getRows() as $row) { $conversions = $row->getColumn($columnNbConversions); if ($conversions > 0 && $count < self::COUNT_TOP_ROWS_TO_DISPLAY // Don't put the "Keyword not defined" in the best segment since it's irritating && !($dimensionName == 'keyword' && $row->getColumn('label') == $keywordNotDefinedString) ) { $topDimension[] = array( 'name' => $row->getColumn('label'), 'nb_conversions' => $conversions, 'conversion_rate' => $this->formatConversionRate($row->getColumn($columnConversionRate)), 'metadata' => $row->getMetadata(), ); $count++; } } $topDimensions[$dimensionName] = $topDimension; } return $topDimensions; } protected function getMetricsForGoal($idGoal) { $request = new Request("method=Goals.get&format=original&idGoal=$idGoal"); $datatable = $request->process(); $dataRow = $datatable->getFirstRow(); $nbConversions = $dataRow->getColumn('nb_conversions'); $nbVisitsConverted = $dataRow->getColumn('nb_visits_converted'); // Backward compatibilty before 1.3, this value was not processed if (empty($nbVisitsConverted)) { $nbVisitsConverted = $nbConversions; } $revenue = $dataRow->getColumn('revenue'); $return = array( 'id' => $idGoal, 'nb_conversions' => (int)$nbConversions, 'nb_visits_converted' => (int)$nbVisitsConverted, 'conversion_rate' => $this->formatConversionRate($dataRow->getColumn('conversion_rate')), 'revenue' => $revenue ? $revenue : 0, 'urlSparklineConversions' => $this->getUrlSparkline('getEvolutionGraph', array('columns' => array('nb_conversions'), 'idGoal' => $idGoal)), 'urlSparklineConversionRate' => $this->getUrlSparkline('getEvolutionGraph', array('columns' => array('conversion_rate'), 'idGoal' => $idGoal)), 'urlSparklineRevenue' => $this->getUrlSparkline('getEvolutionGraph', array('columns' => array('revenue'), 'idGoal' => $idGoal)), ); if ($idGoal == Piwik::LABEL_ID_GOAL_IS_ECOMMERCE_ORDER) { $items = $dataRow->getColumn('items'); $aov = $dataRow->getColumn('avg_order_revenue'); $return = array_merge($return, array( 'revenue_subtotal' => $dataRow->getColumn('revenue_subtotal'), 'revenue_tax' => $dataRow->getColumn('revenue_tax'), 'revenue_shipping' => $dataRow->getColumn('revenue_shipping'), 'revenue_discount' => $dataRow->getColumn('revenue_discount'), 'items' => $items ? $items : 0, 'avg_order_revenue' => $aov ? $aov : 0, 'urlSparklinePurchasedProducts' => $this->getUrlSparkline('getEvolutionGraph', array('columns' => array('items'), 'idGoal' => $idGoal)), 'urlSparklineAverageOrderValue' => $this->getUrlSparkline('getEvolutionGraph', array('columns' => array('avg_order_revenue'), 'idGoal' => $idGoal)), )); } return $return; } /** * Utility function that returns HTML that displays Goal information for reports. This * is the HTML that is at the bottom of every goals page. * * @param int $conversions The number of conversions for this goal (or all goals * in case of the overview). * @param bool $ecommerce Whether to show ecommerce reports or not. * @param bool $cartNbConversions Whether there are cart conversions or not for this * goal. * @return string */ private function getGoalReportsByDimensionTable($conversions, $ecommerce = false, $cartNbConversions = false) { $preloadAbandonedCart = $cartNbConversions !== false && $conversions == 0; $goalReportsByDimension = new ReportsByDimension('Goals'); // add ecommerce reports $ecommerceCustomParams = array(); if ($ecommerce) { if ($preloadAbandonedCart) { $ecommerceCustomParams['abandonedCarts'] = '1'; } else { $ecommerceCustomParams['abandonedCarts'] = '0'; } } if ($conversions > 0 || $ecommerce) { // for non-Goals reports, we show the goals table $customParams = $ecommerceCustomParams + array('documentationForGoalsPage' => '1'); if (Common::getRequestVar('idGoal', '') === '') // if no idGoal, use 0 for overview { $customParams['idGoal'] = '0'; // NOTE: Must be string! Otherwise Piwik_View_HtmlTable_Goals fails. } $allReports = Goals::getReportsWithGoalMetrics(); foreach ($allReports as $category => $reports) { if ($ecommerce) { $categoryText = $this->translationHelper->translateEcommerceMetricCategory($category); } else { $categoryText = $this->translationHelper->translateGoalMetricCategory($category); } foreach ($reports as $report) { if (empty($report['viewDataTable']) && empty($report['abandonedCarts']) ) { $report['viewDataTable'] = 'tableGoals'; } $customParams['viewDataTable'] = $report['viewDataTable']; $goalReportsByDimension->addReport( $categoryText, $report['name'], $report['module'] . '.' . $report['action'], $customParams); } } } return $goalReportsByDimension->render(); } private function setEditGoalsViewVariables($view) { $goals = $this->goals; $view->goals = $goals; $idGoal = Common::getRequestVar('idGoal', 0, 'int'); $view->idGoal = 0; if ($idGoal && array_key_exists($idGoal, $goals)) { $view->idGoal = $idGoal; } // unsanitize goal names and other text data (not done in API so as not to break // any other code/cause security issues) foreach ($goals as &$goal) { $goal['name'] = Common::unsanitizeInputValue($goal['name']); if (isset($goal['pattern'])) { $goal['pattern'] = Common::unsanitizeInputValue($goal['pattern']); } } $view->goalsJSON = json_encode($goals); $view->ecommerceEnabled = $this->site->isEcommerceEnabled(); } private function setUserCanEditGoals(View $view) { $view->userCanEditGoals = Piwik::isUserHasAdminAccess($this->idSite); } }