source: spip-zone/_core_/plugins/medias/action/editer_document.php @ 119908

Last change on this file since 119908 was 119908, checked in by nicod_, 2 months ago

Ne pas considérer une rubrique comme publiée quand on lui ajoute un logo.
https://core.spip.net/issues/4424

File size: 10.2 KB
Line 
1<?php
2
3/***************************************************************************\
4 *  SPIP, Systeme de publication pour l'internet                           *
5 *                                                                         *
6 *  Copyright (c) 2001-2020                                                *
7 *  Arnaud Martin, Antoine Pitrou, Philippe Riviere, Emmanuel Saint-James  *
8 *                                                                         *
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\***************************************************************************/
12
13if (!defined('_ECRIRE_INC_VERSION')) {
14        return;
15}
16
17
18/**
19 * Action editer_document
20 *
21 * @param int $arg
22 * @return array
23 */
24function action_editer_document_dist($arg = null) {
25
26        if (is_null($arg)) {
27                $securiser_action = charger_fonction('securiser_action', 'inc');
28                $arg = $securiser_action();
29        }
30
31        // Envoi depuis le formulaire de creation d'un document
32        if (!$id_document = intval($arg)) {
33                $id_document = document_inserer();
34        }
35
36        if (!$id_document) {
37                return array(0, '');
38        } // erreur
39
40        $err = document_modifier($id_document);
41
42        return array($id_document, $err);
43}
44
45/**
46 * Creer un nouveau document
47 *
48 * @param int $id_parent
49 *     inutilise, pas de parent pour les documents
50 * @param array|null $set
51 * @return int
52 */
53function document_inserer($id_parent = null, $set = null) {
54
55        $champs = array(
56                'statut' => 'prop',
57                'date' => 'NOW()',
58        );
59
60        if ($set) {
61                $champs = array_merge($champs, $set);
62        }
63
64        // Envoyer aux plugins
65        $champs = pipeline(
66                'pre_insertion',
67                array(
68                        'args' => array(
69                                'table' => 'spip_documents',
70                        ),
71                        'data' => $champs
72                )
73        );
74        $id_document = sql_insertq('spip_documents', $champs);
75        pipeline(
76                'post_insertion',
77                array(
78                        'args' => array(
79                                'table' => 'spip_documents',
80                                'id_objet' => $id_document
81                        ),
82                        'data' => $champs
83                )
84        );
85
86        return $id_document;
87}
88
89
90/**
91 * Enregistre une revision de document.
92 * $set est un contenu (par defaut on prend le contenu via _request())
93 *
94 * @param int $id_document
95 * @param array|null $set
96 * @return string|null
97 */
98function document_modifier($id_document, $set = null) {
99
100        include_spip('inc/modifier');
101        include_spip('inc/filtres');
102
103        // champs normaux
104        $champs = collecter_requests(
105                // white list
106                objet_info('document', 'champs_editables'),
107                // black list
108                array('parents', 'ajout_parents'),
109                // donnees eventuellement fournies
110                $set
111        );
112
113
114        $invalideur = '';
115        $indexation = false;
116
117        // Si le document est publie, invalider les caches et demander sa reindexation
118        $t = sql_getfetsel('statut', 'spip_documents', 'id_document=' . intval($id_document));
119        if ($t == 'publie') {
120                $invalideur = "id='id_document/$id_document'";
121                $indexation = true;
122        }
123
124        $ancien_fichier = '';
125        // si le fichier est modifie, noter le nom de l'ancien pour faire le menage
126        if (isset($champs['fichier'])) {
127                $ancien_fichier = sql_getfetsel('fichier', 'spip_documents', 'id_document=' . intval($id_document));
128        }
129
130        if ($err = objet_modifier_champs(
131                'document',
132                $id_document,
133                array(
134                        'data' => $set,
135                        'invalideur' => $invalideur,
136                        'indexation' => $indexation
137                ),
138                $champs
139        )) {
140                return $err;
141        }
142
143        // nettoyer l'ancien fichier si necessaire
144        if (isset($champs['fichier']) // un plugin a pu interdire la modif du fichier en virant le champ
145                and $champs['fichier']
146                and $ancien_fichier // on avait bien note le nom du fichier avant la modif
147                and $ancien_fichier !== $champs['fichier'] // et il a ete modifie
148                and !tester_url_absolue($ancien_fichier)
149                and @file_exists($f = get_spip_doc($ancien_fichier))
150        ) {
151                spip_unlink($f);
152        }
153
154        // Changer le statut du document ?
155        // le statut n'est jamais fixe manuellement mais decoule de celui des objets lies
156        $champs = collecter_requests(array('parents', 'ajouts_parents'), array(), $set);
157        if (document_instituer($id_document, $champs)) {
158                //
159                // Post-modifications
160                //
161
162                // Invalider les caches
163                include_spip('inc/invalideur');
164                suivre_invalideur("id='id_document/$id_document'");
165        }
166}
167
168
169/**
170 * determiner le statut d'un document : prepa/publie
171 * si on trouve un element joint sans champ statut ou avec un statut='publie' alors le doc est publie aussi
172 *
173 * @param int $id_document
174 * @param array $champs
175 * @return bool
176 */
177function document_instituer($id_document, $champs = array()) {
178
179        $statut = isset($champs['statut']) ? $champs['statut'] : null;
180        $date_publication = isset($champs['date_publication']) ? $champs['date_publication'] : null;
181        if (isset($champs['parents'])) {
182                medias_revision_document_parents($id_document, $champs['parents']);
183        }
184        if (isset($champs['ajout_parents'])) {
185                medias_revision_document_parents($id_document, $champs['ajout_parents'], true);
186        }
187
188        $row = sql_fetsel('statut,date_publication', 'spip_documents', 'id_document='.intval($id_document));
189        $statut_ancien = $row['statut'];
190        $date_publication_ancienne = $row['date_publication'];
191
192        /* Autodetermination du statut si non fourni */
193        if (is_null($statut)) {
194                $statut = 'prepa';
195
196                $trouver_table = charger_fonction('trouver_table', 'base');
197                $res = sql_select(
198                        'id_objet,objet, mode',
199                        'spip_documents_liens join spip_documents using(id_document)',
200                        "objet!='document' AND id_document=" . intval($id_document)
201                );
202                // On aura 19 jours 3h14 et 7 secondes pour corriger en 2038 (limitation de la représentation POSIX du temps sur les 32 bits)
203                $date_publication = strtotime('2038-01-01 00:00:00');
204                include_spip('base/objets');
205                while ($row = sql_fetch($res)) {
206                        if (
207                                // si ce n'est pas un logo
208                                !in_array($row['mode'], array('logoon','logooff'))
209                                and (
210                                        // cas particulier des rubriques qui sont publiees des qu'elles contiennent un document !
211                                        $row['objet'] == 'rubrique'
212                                        // ou si objet publie selon sa declaration
213                                        or objet_test_si_publie($row['objet'], $row['id_objet'])
214                                )
215                        ) {
216                                $statut = 'publie';
217                                $date_publication = 0;
218                                continue;
219                        } // si pas publie, et article, il faut checker la date de post-publi eventuelle
220                        elseif ($row['objet'] == 'article'
221                                and $row2 = sql_fetsel(
222                                        'date',
223                                        'spip_articles',
224                                        'id_article=' . intval($row['id_objet']) . " AND statut='publie'"
225                                )
226                        ) {
227                                $statut = 'publie';
228                                $date_publication = min($date_publication, strtotime($row2['date']));
229                        }
230                }
231                $date_publication = date('Y-m-d H:i:s', $date_publication);
232                if ($statut == 'publie' and $statut_ancien == 'publie' and $date_publication == $date_publication_ancienne) {
233                        return false;
234                }
235                if ($statut != 'publie' and $statut_ancien != 'publie' and $statut_ancien != '0') {
236                        return false;
237                }
238        }
239        if ($statut !== $statut_ancien
240                or $date_publication != $date_publication_ancienne
241        ) {
242                sql_updateq(
243                        'spip_documents',
244                        array('statut' => $statut, 'date_publication' => $date_publication),
245                        'id_document=' . intval($id_document)
246                );
247                if ($statut !== $statut_ancien) {
248                        $publier_rubriques = sql_allfetsel(
249                                'id_objet',
250                                'spip_documents_liens',
251                                "objet='rubrique' AND id_document=" . intval($id_document)
252                        );
253                        if (count($publier_rubriques)) {
254                                include_spip('inc/rubriques');
255                                foreach ($publier_rubriques as $r) {
256                                        calculer_rubriques_if($r['id_objet'], array('statut' => $statut), $statut_ancien, false);
257                                }
258                        }
259                }
260                return true;
261        }
262        return false;
263}
264
265
266/**
267 * Revision des parents d'un document
268 * chaque parent est liste au format objet|id_objet
269 *
270 * @param int $id_document
271 * @param array $parents
272 * @param bool $ajout
273 */
274function medias_revision_document_parents($id_document, $parents = null, $ajout = false) {
275        include_spip('inc/autoriser');
276
277        if (!is_array($parents)) {
278                return;
279        }
280
281        $insertions = array();
282        $objets_parents = array(); // array('article'=>array(12,23))
283
284        // au format objet|id_objet
285        foreach ($parents as $p) {
286                $p = explode('|', $p);
287                if (preg_match('/^[a-z0-9_]+$/i', $objet = $p[0])
288                        and (($p[1] = intval($p[1])) or in_array($objet, ['site', 'rubrique']))
289                ) { // securite
290                        $objets_parents[$p[0]][] = $p[1];
291                }
292        }
293
294        include_spip('action/editer_liens');
295        // les liens actuels
296        $liens = objet_trouver_liens(array('document' => $id_document), '*');
297        $deja_parents = array();
298        // si ce n'est pas un ajout, il faut supprimer les liens actuels qui ne sont pas dans $objets_parents
299        if (!$ajout) {
300                foreach ($liens as $k => $lien) {
301                        if (!isset($objets_parents[$lien['objet']]) or !in_array($lien['id_objet'], $objets_parents[$lien['objet']])) {
302                                if (autoriser('dissocierdocuments', $lien['objet'], $lien['id_objet'])) {
303                                        objet_dissocier(array('document' => $id_document), array($lien['objet'] => $lien['id_objet']));
304                                }
305                                unset($liens[$k]);
306                        } else {
307                                $deja_parents[$lien['objet']][] = $lien['id_objet'];
308                        }
309                }
310        }
311
312        // trier les objets à traiter : ne pas prendre en compte ceux qui sont déjà associés ou qu'on n'a pas le droit d'associer
313        foreach ($objets_parents as $objet => $ids) {
314                foreach ($ids as $k => $id) {
315                        if ((
316                                        isset($deja_parents[$objet])
317                                        and in_array($id, $deja_parents[$objet])
318                                )
319                                or !autoriser('associerdocuments', $objet, $id)
320                        ) {
321                                unset($objets_parents[$objet][$k]);
322                        }
323                }
324        }
325        objet_associer(array('document' => $id_document), $objets_parents);
326}
327
328
329// Fonctions Dépréciées
330// --------------------
331
332/**
333 * Insertion d'un document
334 *
335 * @deprecated Utiliser document_inserer()
336 * @see document_inserer()
337 * @return int Identifiant du nouveau document
338 */
339function insert_document() {
340        return document_inserer();
341}
342
343/**
344 * Modification d'un document
345 *
346 * @deprecated Utiliser document_modifier()
347 * @see document_modifier()
348 * @param int $id_document Identifiant du document
349 * @param array|bool $set
350 */
351function document_set($id_document, $set = false) {
352        return document_modifier($id_document, $set);
353}
354
355/**
356 * Insituer un document
357 *
358 * @deprecated Utiliser document_instituer()
359 * @see document_instituer()
360 * @param int $id_document Identifiant du document
361 * @param array $champs
362 */
363function instituer_document($id_document, $champs = array()) {
364        return document_instituer($id_document, $champs);
365}
366
367/**
368 * Réviser un document
369 *
370 * @deprecated Utiliser document_modifier()
371 * @see document_modifier()
372 * @param int $id_document Identifiant du document
373 * @param array $c
374 */
375function revision_document($id_document, $c = false) {
376        return document_modifier($id_document, $c);
377}
Note: See TracBrowser for help on using the repository browser.