Merge "resourceloader: Preload WikiModule::getTitleInfo() in batches"
[lhc/web/wiklou.git] / includes / specials / SpecialChangeContentModel.php
1 <?php
2
3 class SpecialChangeContentModel extends FormSpecialPage {
4
5 public function __construct() {
6 parent::__construct( 'ChangeContentModel', 'editcontentmodel' );
7 }
8
9 public function doesWrites() {
10 return true;
11 }
12
13 /**
14 * @var Title|null
15 */
16 private $title;
17
18 /**
19 * @var Revision|bool|null
20 *
21 * A Revision object, false if no revision exists, null if not loaded yet
22 */
23 private $oldRevision;
24
25 protected function setParameter( $par ) {
26 $par = $this->getRequest()->getVal( 'pagetitle', $par );
27 $title = Title::newFromText( $par );
28 if ( $title ) {
29 $this->title = $title;
30 $this->par = $title->getPrefixedText();
31 } else {
32 $this->par = '';
33 }
34 }
35
36 protected function getDisplayFormat() {
37 return 'ooui';
38 }
39
40 protected function alterForm( HTMLForm $form ) {
41 if ( !$this->title ) {
42 $form->setMethod( 'GET' );
43 }
44
45 $this->addHelpLink( 'Help:ChangeContentModel' );
46
47 // T120576
48 $form->setSubmitTextMsg( 'changecontentmodel-submit' );
49 }
50
51 public function validateTitle( $title ) {
52 if ( !$title ) {
53 // No form input yet
54 return true;
55 }
56
57 // Already validated by HTMLForm, but if not, throw
58 // and exception instead of a fatal
59 $titleObj = Title::newFromTextThrow( $title );
60
61 $this->oldRevision = Revision::newFromTitle( $titleObj ) ?: false;
62
63 if ( $this->oldRevision ) {
64 $oldContent = $this->oldRevision->getContent();
65 if ( !$oldContent->getContentHandler()->supportsDirectEditing() ) {
66 return $this->msg( 'changecontentmodel-nodirectediting' )
67 ->params( ContentHandler::getLocalizedName( $oldContent->getModel() ) )
68 ->escaped();
69 }
70 }
71
72 return true;
73 }
74
75 protected function getFormFields() {
76 $fields = [
77 'pagetitle' => [
78 'type' => 'title',
79 'creatable' => true,
80 'name' => 'pagetitle',
81 'default' => $this->par,
82 'label-message' => 'changecontentmodel-title-label',
83 'validation-callback' => [ $this, 'validateTitle' ],
84 ],
85 ];
86 if ( $this->title ) {
87 $options = $this->getOptionsForTitle( $this->title );
88 if ( empty( $options ) ) {
89 throw new ErrorPageError(
90 'changecontentmodel-emptymodels-title',
91 'changecontentmodel-emptymodels-text',
92 $this->title->getPrefixedText()
93 );
94 }
95 $fields['pagetitle']['readonly'] = true;
96 $fields += [
97 'model' => [
98 'type' => 'select',
99 'name' => 'model',
100 'options' => $options,
101 'label-message' => 'changecontentmodel-model-label'
102 ],
103 'reason' => [
104 'type' => 'text',
105 'name' => 'reason',
106 'validation-callback' => function( $reason ) {
107 $match = EditPage::matchSummarySpamRegex( $reason );
108 if ( $match ) {
109 return $this->msg( 'spamprotectionmatch', $match )->parse();
110 }
111
112 return true;
113 },
114 'label-message' => 'changecontentmodel-reason-label',
115 ],
116 ];
117 }
118
119 return $fields;
120 }
121
122 private function getOptionsForTitle( Title $title = null ) {
123 $models = ContentHandler::getContentModels();
124 $options = [];
125 foreach ( $models as $model ) {
126 $handler = ContentHandler::getForModelID( $model );
127 if ( !$handler->supportsDirectEditing() ) {
128 continue;
129 }
130 if ( $title ) {
131 if ( $title->getContentModel() === $model ) {
132 continue;
133 }
134 if ( !$handler->canBeUsedOn( $title ) ) {
135 continue;
136 }
137 }
138 $options[ContentHandler::getLocalizedName( $model )] = $model;
139 }
140
141 return $options;
142 }
143
144 public function onSubmit( array $data ) {
145 global $wgContLang;
146
147 if ( $data['pagetitle'] === '' ) {
148 // Initial form view of special page, pass
149 return false;
150 }
151
152 // At this point, it has to be a POST request. This is enforced by HTMLForm,
153 // but lets be safe verify that.
154 if ( !$this->getRequest()->wasPosted() ) {
155 throw new RuntimeException( "Form submission was not POSTed" );
156 }
157
158 $this->title = Title::newFromText( $data['pagetitle'] );
159 $user = $this->getUser();
160 // Check permissions and make sure the user has permission to edit the specific page
161 $errors = $this->title->getUserPermissionsErrors( 'editcontentmodel', $user );
162 $errors = wfMergeErrorArrays( $errors, $this->title->getUserPermissionsErrors( 'edit', $user ) );
163 if ( $errors ) {
164 $out = $this->getOutput();
165 $wikitext = $out->formatPermissionsErrorMessage( $errors );
166 // Hack to get our wikitext parsed
167 return Status::newFatal( new RawMessage( '$1', [ $wikitext ] ) );
168 }
169
170 $page = WikiPage::factory( $this->title );
171 if ( $this->oldRevision === null ) {
172 $this->oldRevision = $page->getRevision() ?: false;
173 }
174 $oldModel = $this->title->getContentModel();
175 if ( $this->oldRevision ) {
176 $oldContent = $this->oldRevision->getContent();
177 try {
178 $newContent = ContentHandler::makeContent(
179 $oldContent->getNativeData(), $this->title, $data['model']
180 );
181 } catch ( MWException $e ) {
182 return Status::newFatal(
183 $this->msg( 'changecontentmodel-cannot-convert' )
184 ->params(
185 $this->title->getPrefixedText(),
186 ContentHandler::getLocalizedName( $data['model'] )
187 )
188 );
189 }
190 } else {
191 // Page doesn't exist, create an empty content object
192 $newContent = ContentHandler::getForModelID( $data['model'] )->makeEmptyContent();
193 }
194
195 // All other checks have passed, let's check rate limits
196 if ( $user->pingLimiter( 'editcontentmodel' ) ) {
197 throw new ThrottledError();
198 }
199
200 $flags = $this->oldRevision ? EDIT_UPDATE : EDIT_NEW;
201 $flags |= EDIT_INTERNAL;
202 if ( $user->isAllowed( 'bot' ) ) {
203 $flags |= EDIT_FORCE_BOT;
204 }
205
206 $log = new ManualLogEntry( 'contentmodel', $this->oldRevision ? 'change' : 'new' );
207 $log->setPerformer( $user );
208 $log->setTarget( $this->title );
209 $log->setComment( $data['reason'] );
210 $log->setParameters( [
211 '4::oldmodel' => $oldModel,
212 '5::newmodel' => $data['model']
213 ] );
214
215 $formatter = LogFormatter::newFromEntry( $log );
216 $formatter->setContext( RequestContext::newExtraneousContext( $this->title ) );
217 $reason = $formatter->getPlainActionText();
218 if ( $data['reason'] !== '' ) {
219 $reason .= $this->msg( 'colon-separator' )->inContentLanguage()->text() . $data['reason'];
220 }
221 # Truncate for whole multibyte characters.
222 $reason = $wgContLang->truncate( $reason, 255 );
223
224 $status = $page->doEditContent(
225 $newContent,
226 $reason,
227 $flags,
228 $this->oldRevision ? $this->oldRevision->getId() : false,
229 $user
230 );
231 if ( !$status->isOK() ) {
232 return $status;
233 }
234
235 $logid = $log->insert();
236 $log->publish( $logid );
237
238 return $status;
239 }
240
241 public function onSuccess() {
242 $out = $this->getOutput();
243 $out->setPageTitle( $this->msg( 'changecontentmodel-success-title' ) );
244 $out->addWikiMsg( 'changecontentmodel-success-text', $this->title );
245 }
246
247 /**
248 * Return an array of subpages beginning with $search that this special page will accept.
249 *
250 * @param string $search Prefix to search for
251 * @param int $limit Maximum number of results to return (usually 10)
252 * @param int $offset Number of results to skip (usually 0)
253 * @return string[] Matching subpages
254 */
255 public function prefixSearchSubpages( $search, $limit, $offset ) {
256 return $this->prefixSearchString( $search, $limit, $offset );
257 }
258
259 protected function getGroupName() {
260 return 'pagetools';
261 }
262 }