3 /***************************************************************************\
4 * SPIP, Systeme de publication pour l'internet *
6 * Copyright (c) 2001-2017 *
7 * Arnaud Martin, Antoine Pitrou, Philippe Riviere, Emmanuel Saint-James *
9 * Ce programme est un logiciel libre distribue sous licence GNU/GPL. *
10 * Pour plus de details voir le fichier COPYING.txt ou l'aide en ligne. *
11 \***************************************************************************/
14 * Gestion de l'action editer_article et de l'API d'édition d'un article
16 * @package SPIP\Core\Articles\Edition
19 if (!defined('_ECRIRE_INC_VERSION')) {
24 * Action d'édition d'un article dans la base de données dont
25 * l'identifiant est donné en paramètre de cette fonction ou
26 * en argument de l'action sécurisée
28 * Si aucun identifiant n'est donné, on crée alors un nouvel article,
29 * à condition que la rubrique parente (id_rubrique) puisse être obtenue
32 * @link http://code.spip.net/@action_editer_article_dist
33 * @uses article_inserer()
34 * @uses article_modifier()
36 * @param null|int $arg
37 * Identifiant de l'article. En absence utilise l'argument
38 * de l'action sécurisée.
40 * Liste (identifiant de l'article, Texte d'erreur éventuel)
42 function action_editer_article_dist($arg = null) {
43 include_spip('inc/autoriser');
46 $securiser_action = charger_fonction('securiser_action', 'inc');
47 $arg = $securiser_action();
50 // si id_article n'est pas un nombre, c'est une creation
51 // mais on verifie qu'on a toutes les donnees qu'il faut.
52 if (!$id_article = intval($arg)) {
53 $id_parent = _request('id_parent');
55 $err = _L("creation interdite d'un article sans rubrique");
56 } elseif (!autoriser('creerarticledans', 'rubrique', $id_parent)) {
57 $err = _T("info_creerdansrubrique_non_autorise");
59 $id_article = article_inserer($id_parent);
63 // Enregistre l'envoi dans la BD
64 if ($id_article > 0) {
65 $err = article_modifier($id_article);
69 spip_log("echec editeur article: $err", _LOG_ERREUR
);
72 return array($id_article, $err);
78 * Appelle toutes les fonctions de modification d'un article
80 * @param int $id_article
81 * Identifiant de l'article à modifier
82 * @param array|null $set
83 * Couples (colonne => valeur) de données à modifier.
84 * En leur absence, on cherche les données dans les champs éditables
85 * qui ont été postés (via collecter_requests())
88 * - Chaîne vide si aucune erreur,
89 * - Null si aucun champ à modifier,
90 * - Chaîne contenant un texte d'erreur sinon.
92 function article_modifier($id_article, $set = null) {
94 // unifier $texte en cas de texte trop long
95 trop_longs_articles();
97 include_spip('inc/modifier');
98 include_spip('inc/filtres');
99 $c = collecter_requests(
101 objet_info('article', 'champs_editables'),
103 array('date', 'statut', 'id_parent'),
104 // donnees eventuellement fournies
108 // Si l'article est publie, invalider les caches et demander sa reindexation
109 $t = sql_getfetsel("statut", "spip_articles", "id_article=" . intval($id_article));
110 $invalideur = $indexation = false;
111 if ($t == 'publie') {
112 $invalideur = "id='article/$id_article'";
116 if ($err = objet_modifier_champs('article', $id_article,
119 'nonvide' => array('titre' => _T('info_nouvel_article') . " " . _T('info_numero_abbreviation') . $id_article),
120 'invalideur' => $invalideur,
121 'indexation' => $indexation,
122 'date_modif' => 'date_modif' // champ a mettre a date('Y-m-d H:i:s') s'il y a modif
129 // Modification de statut, changement de rubrique ?
130 $c = collecter_requests(array('date', 'statut', 'id_parent'), array(), $set);
131 $err = article_instituer($id_article, $c);
137 * Insérer un nouvel article en base de données
139 * En plus des données enregistrées par défaut, la fonction :
141 * - retrouve un identifiant de rubrique pour stocker l'article (la
142 * première rubrique racine) si l'identifiant de rubrique transmis est
144 * - calcule la langue de l'article, soit
145 * - d'après la langue de la rubrique si les articles ne sont pas
146 * configurés comme pouvant être traduits,
147 * - d'après la langue de l'auteur en cours si les articles peuvent être traduits et
148 * si la langue de l'auteur est acceptée en tant que langue de traduction
149 * - crée une liaison automatiquement entre l'auteur connecté et l'article
150 * créé, de sorte que la personne devient par défaut auteur de l'article
153 * @pipeline_appel pre_insertion
154 * @pipeline_appel post_insertion
157 * @global array visiteur_session
158 * @global string spip_lang
160 * @param int $id_rubrique
161 * Identifiant de la rubrique parente
162 * @param array|null $set
164 * Identifiant du nouvel article
167 function article_inserer($id_rubrique, $set = null) {
169 // Si id_rubrique vaut 0 ou n'est pas definie, creer l'article
170 // dans la premiere rubrique racine
171 if (!$id_rubrique = intval($id_rubrique)) {
172 $row = sql_fetsel("id_rubrique, id_secteur, lang", "spip_rubriques", "id_parent=0", '', '0+titre,titre', "1");
173 $id_rubrique = $row['id_rubrique'];
175 $row = sql_fetsel("lang, id_secteur", "spip_rubriques", "id_rubrique=$id_rubrique");
178 // eviter $id_secteur = NULL (erreur sqlite) si la requete precedente echoue
179 // cas de id_rubrique = -1 par exemple avec plugin "pages"
180 $id_secteur = isset($row['id_secteur']) ?
$row['id_secteur'] : 0;
182 $lang_rub = $row['lang'];
186 // La langue a la creation : si les liens de traduction sont autorises
187 // dans les rubriques, on essaie avec la langue de l'auteur,
188 // ou a defaut celle de la rubrique
189 // Sinon c'est la langue de la rubrique qui est choisie + heritee
190 if (!empty($GLOBALS['meta']['multi_objets']) and in_array('spip_articles',
191 explode(',', $GLOBALS['meta']['multi_objets']))
193 lang_select($GLOBALS['visiteur_session']['lang']);
194 if (in_array($GLOBALS['spip_lang'],
195 explode(',', $GLOBALS['meta']['langues_multilingue']))) {
196 $lang = $GLOBALS['spip_lang'];
203 $lang = $lang_rub ?
$lang_rub : $GLOBALS['meta']['langue_site'];
207 'id_rubrique' => $id_rubrique,
208 'id_secteur' => $id_secteur,
210 'date' => date('Y-m-d H:i:s'),
212 'langue_choisie' => $choisie
216 $champs = array_merge($champs, $set);
219 // Envoyer aux plugins
220 $champs = pipeline('pre_insertion',
223 'table' => 'spip_articles',
229 $id_article = sql_insertq("spip_articles", $champs);
231 // controler si le serveur n'a pas renvoye une erreur
232 if ($id_article > 0) {
233 $id_auteur = ((is_null(_request('id_auteur')) and isset($GLOBALS['visiteur_session']['id_auteur'])) ?
234 $GLOBALS['visiteur_session']['id_auteur']
235 : _request('id_auteur'));
237 include_spip('action/editer_auteur');
238 auteur_associer($id_auteur, array('article' => $id_article));
242 pipeline('post_insertion',
245 'table' => 'spip_articles',
246 'id_objet' => $id_article
257 * Modification des statuts d'un article
259 * Modifie la langue, la rubrique ou les statuts d'un article.
261 * @global array $GLOBALS ['meta']
263 * @pipeline_appel pre_edition
264 * @pipeline_appel post_edition
266 * @param int $id_article
267 * Identifiant de l'article
269 * Couples (colonne => valeur) des données à instituer
270 * Les colonnes 'statut' et 'id_parent' sont liées, car un admin restreint
271 * peut deplacer un article publié vers une rubrique qu'il n'administre pas
272 * @param bool $calcul_rub
273 * True pour changer le statut des rubriques concernées si un article
274 * change de statut ou est déplacé dans une autre rubrique
278 function article_instituer($id_article, $c, $calcul_rub = true) {
280 include_spip('inc/autoriser');
281 include_spip('inc/rubriques');
282 include_spip('inc/modifier');
284 $row = sql_fetsel("statut, date, id_rubrique", "spip_articles", "id_article=$id_article");
285 $id_rubrique = $row['id_rubrique'];
286 $statut_ancien = $statut = $row['statut'];
287 $date_ancienne = $date = $row['date'];
290 $d = isset($c['date']) ?
$c['date'] : null;
291 $s = isset($c['statut']) ?
$c['statut'] : $statut;
293 // cf autorisations dans inc/instituer_article
294 if ($s != $statut or ($d and $d != $date)) {
295 if (autoriser('publierdans', 'rubrique', $id_rubrique)) {
296 $statut = $champs['statut'] = $s;
298 if (autoriser('modifier', 'article', $id_article) and $s != 'publie') {
299 $statut = $champs['statut'] = $s;
301 spip_log("editer_article $id_article refus " . join(' ', $c));
305 // En cas de publication, fixer la date a "maintenant"
306 // sauf si $c commande autre chose
307 // ou si l'article est deja date dans le futur
308 // En cas de proposition d'un article (mais pas depublication), idem
309 if ($champs['statut'] == 'publie'
310 or ($champs['statut'] == 'prop' and ($d or !in_array($statut_ancien, array('publie', 'prop'))))
312 if ($d or strtotime($d = $date) > time()) {
313 $champs['date'] = $date = $d;
315 $champs['date'] = $date = date('Y-m-d H:i:s');
320 // Verifier que la rubrique demandee existe et est differente
321 // de la rubrique actuelle
322 if (isset($c['id_parent'])
323 and $id_parent = $c['id_parent']
324 and $id_parent != $id_rubrique
325 and (sql_fetsel('1', "spip_rubriques", "id_rubrique=" . intval($id_parent)))
327 $champs['id_rubrique'] = $id_parent;
329 // si l'article etait publie
330 // et que le demandeur n'est pas admin de la rubrique de destination
331 // repasser l'article en statut 'propose'.
332 if ($statut == 'publie'
333 and !autoriser('publierdans', 'rubrique', $id_parent)
335 $champs['statut'] = 'prop';
339 // Envoyer aux plugins
340 $champs = pipeline('pre_edition',
343 'table' => 'spip_articles',
344 'id_objet' => $id_article,
345 'action' => 'instituer',
346 'statut_ancien' => $statut_ancien,
347 'date_ancienne' => $date_ancienne,
353 if (!count($champs)) {
357 // Envoyer les modifs.
358 editer_article_heritage($id_article, $id_rubrique, $statut_ancien, $champs, $calcul_rub);
360 // Invalider les caches
361 include_spip('inc/invalideur');
362 suivre_invalideur("id='article/$id_article'");
365 $t = strtotime($date);
366 $p = @$GLOBALS['meta']['date_prochain_postdate'];
367 if ($t > time() and (!$p or ($t < $p))) {
368 ecrire_meta('date_prochain_postdate', $t);
373 pipeline('post_edition',
376 'table' => 'spip_articles',
377 'id_objet' => $id_article,
378 'action' => 'instituer',
379 'statut_ancien' => $statut_ancien,
380 'date_ancienne' => $date_ancienne,
387 if ($notifications = charger_fonction('notifications', 'inc')) {
388 $notifications('instituerarticle', $id_article,
389 array('statut' => $statut, 'statut_ancien' => $statut_ancien, 'date' => $date, 'date_ancienne' => $date_ancienne)
393 return ''; // pas d'erreur
397 * Fabrique la requête de modification de l'article, avec champs hérités
399 * @global array $GLOBALS ['meta']
401 * @param int $id_article
402 * Identifiant de l'article
403 * @param int $id_rubrique
404 * Identifiant de la rubrique parente
405 * @param string $statut
406 * Statut de l'article (prop, publie, ...)
407 * @param array $champs
408 * Couples (colonne => valeur) des champs qui ont été modifiés
410 * True pour actualiser le statut et date de publication de la rubrique
411 * parente si nécessaire
413 * null si aucune action à faire
416 function editer_article_heritage($id_article, $id_rubrique, $statut, $champs, $cond = true) {
418 // Si on deplace l'article
419 // changer aussi son secteur et sa langue (si heritee)
420 if (isset($champs['id_rubrique'])) {
422 $row_rub = sql_fetsel("id_secteur, lang", "spip_rubriques", "id_rubrique=" . sql_quote($champs['id_rubrique']));
424 $langue = $row_rub['lang'];
425 $champs['id_secteur'] = $row_rub['id_secteur'];
426 if (sql_fetsel('1', 'spip_articles',
427 "id_article=" . intval($id_article) . " AND langue_choisie<>'oui' AND lang<>" . sql_quote($langue))) {
428 $champs['lang'] = $langue;
436 sql_updateq('spip_articles', $champs, "id_article=" . intval($id_article));
438 // Changer le statut des rubriques concernees
441 include_spip('inc/rubriques');
442 $postdate = ($GLOBALS['meta']["post_dates"] == "non" and isset($champs['date']) and (strtotime($champs['date']) < time())) ?
$champs['date'] : false;
443 calculer_rubriques_if($id_rubrique, $champs, $statut, $postdate);
448 * Réunit les textes decoupés parce que trop longs
452 function trop_longs_articles() {
453 if (is_array($plus = _request('texte_plus'))) {
454 foreach ($plus as $n => $t) {
455 $plus[$n] = preg_replace(",<!--SPIP-->[\n\r]*,", "", $t);
457 set_request('texte', join('', $plus) . _request('texte'));
462 // Fonctions Dépréciées
463 // --------------------
466 * Créer une révision d'un article
468 * @deprecated Utiliser article_modifier()
469 * @see article_modifier()
471 * @param int $id_article
472 * Identifiant de l'article à modifier
473 * @param array|null $c
474 * Couples (colonne => valeur) de données à modifier.
475 * En leur absence, on cherche les données dans les champs éditables
476 * qui ont été postés (via _request())
477 * @return string|null
478 * Chaîne vide si aucune erreur,
479 * Null si aucun champ à modifier,
480 * Chaîne contenant un texte d'erreur sinon.
482 function revisions_articles($id_article, $c = false) {
483 return article_modifier($id_article, $c);
487 * Créer une révision d'un article
489 * @deprecated Utiliser article_modifier()
490 * @see article_modifier()
492 * @param int $id_article
493 * Identifiant de l'article à modifier
494 * @param array|null $c
495 * Couples (colonne => valeur) de données à modifier.
496 * En leur absence, on cherche les données dans les champs éditables
497 * qui ont été postés (via _request())
498 * @return string|null
499 * Chaîne vide si aucune erreur,
500 * Null si aucun champ à modifier,
501 * Chaîne contenant un texte d'erreur sinon.
503 function revision_article($id_article, $c = false) {
504 return article_modifier($id_article, $c);
508 * Modifier un article
510 * @deprecated Utiliser article_modifier()
511 * @see article_modifier()
513 * @param int $id_article
514 * Identifiant de l'article à modifier
515 * @param array|null $set
516 * Couples (colonne => valeur) de données à modifier.
517 * En leur absence, on cherche les données dans les champs éditables
518 * qui ont été postés (via _request())
519 * @return string|null
520 * Chaîne vide si aucune erreur,
521 * Null si aucun champ à modifier,
522 * Chaîne contenant un texte d'erreur sinon.
524 function articles_set($id_article, $set = null) {
525 return article_modifier($id_article, $set);
529 * Insertion d'un article dans une rubrique
531 * @deprecated Utiliser article_inserer()
532 * @see article_inserer()
534 * @param int $id_rubrique
535 * Identifiant de la rubrique
537 * Identifiant du nouvel article
539 function insert_article($id_rubrique) {
540 return article_inserer($id_rubrique);
544 * Instituer un article dans une rubrique
546 * @deprecated Utiliser article_instituer()
547 * @see article_instituer()
549 * @param int $id_article
550 * Identifiant de l'article
552 * Couples (colonne => valeur) des données à instituer
553 * Les colonnes 'statut' et 'id_parent' sont liées, car un admin restreint
554 * peut deplacer un article publié vers une rubrique qu'il n'administre pas
555 * @param bool $calcul_rub
556 * True pour changer le statut des rubriques concernées si un article
557 * change de statut ou est déplacé dans une autre rubrique
561 function instituer_article($id_article, $c, $calcul_rub = true) {
562 return article_instituer($id_article, $c, $calcul_rub);