* @param User $user
* @param LinkTarget $target
*
- * @return WatchedItem|null
+ * @return WatchedItem|false
*/
private function getCached( User $user, LinkTarget $target ) {
return $this->cache->get( $this->getCacheKey( $user, $target ) );
$watchedItems = [];
foreach ( $res as $row ) {
- // todo these could all be cached at some point?
+ // @todo: Should we add these to the process cache?
$watchedItems[] = new WatchedItem(
$user,
new TitleValue( (int)$row->wl_namespace, $row->wl_title ),
}
$rows = [];
+ $items = [];
foreach ( $targets as $target ) {
$rows[] = [
'wl_user' => $user->getId(),
'wl_title' => $target->getDBkey(),
'wl_notificationtimestamp' => null,
];
+ $items[] = new WatchedItem(
+ $user,
+ $target,
+ null
+ );
$this->uncache( $user, $target );
}
// if there's already an entry for this page
$dbw->insert( 'watchlist', $toInsert, __METHOD__, 'IGNORE' );
}
+ // Update process cache to ensure skin doesn't claim that the current
+ // page is unwatched in the response of action=watch itself (T28292).
+ // This would otherwise be re-queried from a slave by isWatched().
+ foreach ( $items as $item ) {
+ $this->cache( $item );
+ }
return true;
}
--- /dev/null
+<?php
+/**
+ * MediaWiki Widgets – DateInputWidget class.
+ *
+ * @copyright 2016 MediaWiki Widgets Team and others; see AUTHORS.txt
+ * @license The MIT License (MIT); see LICENSE.txt
+ */
+
+namespace MediaWiki\Widget;
+
+use DateTime;
+
+/**
+ * Date input widget.
+ *
+ * @since 1.29
+ */
+class DateInputWidget extends \OOUI\TextInputWidget {
+
+ protected $inputFormat = null;
+ protected $displayFormat = null;
+ protected $placeholderLabel = null;
+ protected $placeholderDateFormat = null;
+ protected $precision = null;
+ protected $mustBeAfter = null;
+ protected $mustBeBefore = null;
+ protected $overlay = null;
+
+ /**
+ * @param array $config Configuration options
+ * @param string $config['inputFormat'] Date format string to use for the textual input field.
+ * Displayed while the widget is active, and the user can type in a date in this format.
+ * Should be short and easy to type. (default: 'YYYY-MM-DD' or 'YYYY-MM', depending on
+ * `precision`)
+ * @param string $config['displayFormat'] Date format string to use for the clickable label.
+ * while the widget is inactive. Should be as unambiguous as possible (for example, prefer
+ * to spell out the month, rather than rely on the order), even if that makes it longer.
+ * Applicable only if the widget is infused. (default: language-specific)
+ * @param string $config['placeholderLabel'] Placeholder text shown when the widget is not
+ * selected. Applicable only if the widget is infused. (default: taken from message
+ * `mw-widgets-dateinput-no-date`)
+ * @param string $config['placeholderDateFormat'] User-visible date format string displayed
+ * in the textual input field when it's empty. Should be the same as `inputFormat`, but
+ * translated to the user's language. (default: 'YYYY-MM-DD' or 'YYYY-MM', depending on
+ * `precision`)
+ * @param string $config['precision'] Date precision to use, 'day' or 'month' (default: 'day')
+ * @param string $config['mustBeAfter'] Validates the date to be after this.
+ * In the 'YYYY-MM-DD' or 'YYYY-MM' format, depending on `precision`.
+ * @param string $config['mustBeBefore'] Validates the date to be before this.
+ * In the 'YYYY-MM-DD' or 'YYYY-MM' format, depending on `precision`.
+ * @param string $config['overlay'] The jQuery selector for the overlay layer on which to render
+ * the calendar. This configuration is useful in cases where the expanded calendar is larger
+ * than its container. The specified overlay layer is usually on top of the container and has
+ * a larger area. Applicable only if the widget is infused. By default, the calendar uses
+ * relative positioning.
+ */
+ public function __construct( array $config = [] ) {
+ $config = array_merge( [
+ // Default config values
+ 'precision' => 'day',
+ ], $config );
+
+ // Properties
+ if ( isset( $config['inputFormat'] ) ) {
+ $this->inputFormat = $config['inputFormat'];
+ }
+ if ( isset( $config['placeholderDateFormat'] ) ) {
+ $this->placeholderDateFormat = $config['placeholderDateFormat'];
+ }
+ $this->precision = $config['precision'];
+ if ( isset( $config['mustBeAfter'] ) ) {
+ $this->mustBeAfter = $config['mustBeAfter'];
+ }
+ if ( isset( $config['mustBeBefore'] ) ) {
+ $this->mustBeBefore = $config['mustBeBefore'];
+ }
+
+ // Properties stored for the infused JS widget
+ if ( isset( $config['displayFormat'] ) ) {
+ $this->displayFormat = $config['displayFormat'];
+ }
+ if ( isset( $config['placeholderLabel'] ) ) {
+ $this->placeholderLabel = $config['placeholderLabel'];
+ }
+ if ( isset( $config['overlay'] ) ) {
+ $this->overlay = $config['overlay'];
+ }
+
+ // Set up placeholder text visible if the browser doesn't override it (logic taken from JS)
+ if ( $this->placeholderDateFormat !== null ) {
+ $placeholder = $this->placeholderDateFormat;
+ } elseif ( $this->inputFormat !== null ) {
+ // We have no way to display a translated placeholder for custom formats
+ $placeholder = '';
+ } else {
+ $placeholder = wfMessage( "mw-widgets-dateinput-placeholder-$this->precision" )->text();
+ }
+
+ $config = array_merge( [
+ // Processed config values
+ 'placeholder' => $placeholder,
+ ], $config );
+
+ // Parent constructor
+ parent::__construct( $config );
+
+ // Calculate min/max attributes (which are skipped by TextInputWidget) and add to <input>
+ // min/max attributes are inclusive, but mustBeAfter/Before are exclusive
+ if ( $this->mustBeAfter !== null ) {
+ $min = new DateTime( $this->mustBeAfter );
+ $min = $min->modify( '+1 day' );
+ $min = $min->format( 'Y-m-d' );
+ $this->input->setAttributes( [ 'min' => $min ] );
+ }
+ if ( $this->mustBeBefore !== null ) {
+ $max = new DateTime( $this->mustBeBefore );
+ $max = $max->modify( '-1 day' );
+ $max = $max->format( 'Y-m-d' );
+ $this->input->setAttributes( [ 'max' => $max ] );
+ }
+
+ // Initialization
+ $this->addClasses( [ 'mw-widget-dateInputWidget' ] );
+ }
+
+ protected function getJavaScriptClassName() {
+ return 'mw.widgets.DateInputWidget';
+ }
+
+ public function getConfig( &$config ) {
+ if ( $this->inputFormat !== null ) {
+ $config['inputFormat'] = $this->inputFormat;
+ }
+ if ( $this->displayFormat !== null ) {
+ $config['displayFormat'] = $this->displayFormat;
+ }
+ if ( $this->placeholderLabel !== null ) {
+ $config['placeholderLabel'] = $this->placeholderLabel;
+ }
+ if ( $this->placeholderDateFormat !== null ) {
+ $config['placeholderDateFormat'] = $this->placeholderDateFormat;
+ }
+ if ( $this->precision !== null ) {
+ $config['precision'] = $this->precision;
+ }
+ if ( $this->mustBeAfter !== null ) {
+ $config['mustBeAfter'] = $this->mustBeAfter;
+ }
+ if ( $this->mustBeBefore !== null ) {
+ $config['mustBeBefore'] = $this->mustBeBefore;
+ }
+ if ( $this->overlay !== null ) {
+ $config['overlay'] = $this->overlay;
+ }
+ return parent::getConfig( $config );
+ }
+
+ public function getInputElement( $config ) {
+ // Inserts date/month type attribute
+ return parent::getInputElement( $config )
+ ->setAttributes( [
+ 'type' => ( $config['precision'] === 'month' ) ? 'month' : 'date'
+ ] );
+ }
+}