*/ /** * Special:GadgetUsage - Lists all the gadgets on the wiki along with number of users. * @ingroup SpecialPage */ class SpecialGadgetUsage extends QueryPage { function __construct( $name = 'GadgetUsage' ) { parent::__construct( $name ); $this->limit = 1000; // Show all gadgets $this->shownavigation = false; } public function isExpensive() { return true; } /** * SQL Query being used: * SELECT up_property, SUM(up_value), count(qcc_title) * FROM user_properties * LEFT JOIN user ON up_user = user_id * LEFT JOIN querycachetwo ON user_name = qcc_title AND qcc_type = 'activeusers' AND up_value = 1 * WHERE up_property LIKE 'gadget-%' * GROUP BY up_property; */ public function getQueryInfo() { $dbr = wfGetDB( DB_SLAVE ); return array( 'tables' => array( 'user_properties', 'user', 'querycachetwo' ), 'fields' => array( 'title' => 'up_property', 'value' => 'SUM( up_value )', // Need to pick fields existing in the querycache table so that the results are cachable 'namespace' => 'COUNT( qcc_title )' ), 'conds' => array( 'up_property' . $dbr->buildLike( 'gadget-', $dbr->anyString() ) ), 'options' => array( 'GROUP BY' => array( 'up_property' ) ), 'join_conds' => array( 'user' => array( 'LEFT JOIN', array( 'up_user = user_id' ) ), 'querycachetwo' => array( 'LEFT JOIN', array( 'user_name = qcc_title', 'qcc_type = "activeusers"', 'up_value = 1' ) ) ) ); } public function getOrderFields() { return array( 'value' ); } /** * Output the start of the table * Including opening , and first with column headers. */ protected function outputTableStart() { $html = Html::openElement( 'table', array( 'class' => array( 'sortable', 'wikitable' ) ) ); $html .= Html::openElement( 'tr', array() ); $headers = array( 'gadgetusage-gadget', 'gadgetusage-usercount', 'gadgetusage-activeusers' ); foreach( $headers as $h ) { $html .= Html::element( 'th', array(), $this->msg( $h )->text() ); } $html .= Html::closeElement( 'tr' ); $this->getOutput()->addHTML( $html ); } /** * @param Skin $skin * @param object $result Result row * @return string|bool String of HTML */ public function formatResult( $skin, $result ) { $gadgetTitle = substr( $result->title, 7 ); $gadgetUserCount = $this->getLanguage()->formatNum( $result->value ); $activeUsers = $this->getLanguage()->formatNum( $result->namespace ); if ( $gadgetTitle ) { $html = Html::openElement( 'tr', array() ); $html .= Html::element( 'td', array(), $gadgetTitle ); $html .= Html::element( 'td', array(), $gadgetUserCount ); $html .= Html::element( 'td', array(), $activeUsers ); $html .= Html::closeElement( 'tr' ); return $html; } return false; } /** * Get a list of default gadgets * @param GadgetRepo $gadgetRepo * @param array $gadgetIds list of gagdet ids registered in the wiki * @return array */ protected function getDefaultGadgets( $gadgetRepo, $gadgetIds ) { $gadgetsList = array(); foreach ( $gadgetIds as $g ) { $gadget = $gadgetRepo->getGadget( $g ); if ( $gadget->isOnByDefault() ) { $gadgetsList[] = $gadget->getName(); } } asort( $gadgetsList, SORT_STRING | SORT_FLAG_CASE ); return $gadgetsList; } /** * Format and output report results using the given information plus * OutputPage * * @param OutputPage $out OutputPage to print to * @param Skin $skin User skin to use * @param IDatabase $dbr Database (read) connection to use * @param ResultWrapper $res Result pointer * @param int $num Number of available result rows * @param int $offset Paging offset */ protected function outputResults( $out, $skin, $dbr, $res, $num, $offset ) { $gadgetRepo = GadgetRepo::singleton(); $gadgetIds = $gadgetRepo->getGadgetIds(); $defaultGadgets = $this->getDefaultGadgets( $gadgetRepo, $gadgetIds ); $out->addHtml( $this->msg( 'gadgetusage-intro', $this->getConfig()->get( 'ActiveUserDays' ) )->parseAsBlock() ); if ( $num > 0 ) { $this->outputTableStart(); // Append default gadgets to the table with 'default' in the total and active user fields foreach ( $defaultGadgets as $default ) { $html = Html::openElement( 'tr', array() ); $html .= Html::element( 'td', array(), $default ); $html .= Html::element( 'td', array(), $this->msg( 'gadgetusage-default' ) ); $html .= Html::element( 'td', array(), $this->msg( 'gadgetusage-default' ) ); $html .= Html::closeElement( 'tr' ); $out->addHTML( $html ); } foreach ( $res as $row ) { // Remove the 'gadget-' part of the result string and compare if it's present // in $defaultGadgets, if not we format it and add it to the output if ( !in_array( substr( $row->title, 7 ), $defaultGadgets ) ) { // Only pick gadgets which are in the list $gadgetIds to make sure they exist if ( in_array( substr( $row->title, 7 ), $gadgetIds ) ) { $line = $this->formatResult( $skin, $row ); if ( $line ) { $out->addHTML( $line ); } } } } // Close table element $out->addHtml( Html::closeElement( 'table' ) ); } else { $out->addHtml( $this->msg( 'gadgetusage-noresults' )->parseAsBlock() ); } } protected function getGroupName() { return 'wiki'; } }