node.module 25.6 KB
Newer Older
Dries's avatar
 
Dries committed
1
<?php
2
// $Id$
Dries's avatar
 
Dries committed
3

Dries's avatar
   
Dries committed
4
5
6
7
8
9
function node_help() {
  global $mod;

  if ($mod == "node") {
    foreach (module_list() as $name) {
      if (module_hook($name, "status") && $name != "node") {
Dries's avatar
   
Dries committed
10
        print "<h3>". ucfirst($name) ." type</h3>";
Dries's avatar
   
Dries committed
11
12
13
14
15
16
        print module_invoke($name, "help");
      }
    }
  }
}

Dries's avatar
   
Dries committed
17
18
function node_access($op, $node = 0) {

Dries's avatar
   
Dries committed
19
20
  if (user_access("administer nodes")) {
    return 1;
Dries's avatar
   
Dries committed
21
  }
Dries's avatar
   
Dries committed
22
  else {
Dries's avatar
   
Dries committed
23

Dries's avatar
   
Dries committed
24
25
26
    /*
    ** Convert the node to an object if necessary:
    */
Dries's avatar
   
Dries committed
27

Dries's avatar
   
Dries committed
28
29
30
    if (is_array($node)) {
      $node = node_object($node);
    }
Dries's avatar
   
Dries committed
31

Dries's avatar
   
Dries committed
32
33
34
35
    /*
    ** Construct a function:
    */

Dries's avatar
   
Dries committed
36
37
38
39
40
41
42
43
    if ($node->type) {
      $type = $node->type;
    }
    else {
      $type = $node;
    }

    $function = $type ."_access";
Dries's avatar
   
Dries committed
44
45
46
47
48
49
50

    if (function_exists($function)) {
      return $function($op, $node);
    }
    else {
      return 0;
    }
Dries's avatar
   
Dries committed
51
52
53
  }
}

Dries's avatar
   
Dries committed
54
function node_perm() {
Dries's avatar
   
Dries committed
55
  return array("administer nodes", "access content", "post content");
Dries's avatar
   
Dries committed
56
57
}

Dries's avatar
   
Dries committed
58
function node_search($keys) {
Dries's avatar
   
Dries committed
59
  global $PHP_SELF;
Dries's avatar
   
Dries committed
60
61
62

  $result = db_query("SELECT n.nid, n.title, n.created, u.uid, u.name FROM node n LEFT JOIN users u ON n.uid = u.uid WHERE n.status = 1 AND (n.title LIKE '%$keys%' OR n.teaser LIKE '%$keys%' OR n.body LIKE '%$keys%') ORDER BY n.created DESC LIMIT 20");
  while ($node = db_fetch_object($result)) {
Dries's avatar
   
Dries committed
63
    $find[$i++] = array("title" => check_output($node->title), "link" => (strstr($PHP_SELF, "admin.php") ? "admin.php?mod=node&type=node&op=edit&id=$node->nid" : "node.php?id=$node->nid"), "user" => $node->name, "date" => $node->created);
Dries's avatar
   
Dries committed
64
65
66
67
68
  }

  return $find;
}

Dries's avatar
   
Dries committed
69
function node_conf_options() {
Dries's avatar
   
Dries committed
70
  $output .= form_select(t("Default number of nodes to display"), "default_nodes_main", variable_get("default_nodes_main", 10), array(1 => 1, 2 => 2, 3 => 3, 4 => 4, 5 =>  5, 6 => 6, 7 => 7, 8 => 8, 9 => 9, 10 => 10, 15 => 15, 20 => 20, 25 => 25, 30 => 30), t("The default maximum number of nodes to display on the main page."));
Dries's avatar
   
Dries committed
71
72
73
  return $output;
}

Dries's avatar
CHANGES    
Dries committed
74
function node_conf_filters() {
Dries's avatar
   
Dries committed
75
  $output .= form_select(t("Enable HTML tags"), "filter_html", variable_get("filter_html", 0), array("Disabled", "Enabled"), t("Allow HTML and PHP tags in user-contributed content."));
Dries's avatar
   
Dries committed
76
  $output .= form_textfield(t("Allowed HTML tags"), "allowed_html", variable_get("allowed_html", "<a> <b> <dd> <dl> <dt> <i> <li> <ol> <u> <ul>"), 64, 128, t("If enabled, optionally specify tags which should not be stripped.  'STYLE' attributes, 'ON' attributes and unclosed tags are always stripped."));
Dries's avatar
   
Dries committed
77
  $output .= "<hr />";
Dries's avatar
   
Dries committed
78
  $output .= form_select(t("Enable link tags"), "filter_link", variable_get("filter_link", 0), array("Disabled", "Enabled"), t("Substitute special [[nodesubject|text]] tags. Your browser will display 'text', and when you click on it your browser will open the node with the subject 'nodesubject'. Please be aware that you'll need to copy the subject of the target node exactly in order to use this feature."));
Dries's avatar
   
Dries committed
79
  $output .= "<hr />";
Dries's avatar
CHANGES    
Dries committed
80
81
82
83
84
85
86
87
88
89
90
  return $output;
}

function node_filter_html($text) {
  $text = eregi_replace("([ \f\r\t\n\'\"])style=[^>]+", "\\1", $text);
  $text = eregi_replace("([ \f\r\t\n\'\"])on[a-z]+=[^>]+", "\\1", $text);
  $text = strip_tags($text, variable_get("allowed_html", ""));
  return $text;
}

function node_filter_link($text) {
Dries's avatar
   
Dries committed
91
92
93
94
95
  $src = array("/\[\[(([^\|]*?)(\|([^\|]*?))?)\]\]/e");  // [link|description]
  $dst = array(format_tag('\\2', '\\4'));                // [link|description]
  return preg_replace($src, $dst, $text);
}

Dries's avatar
   
Dries committed
96
97
98
99
100
101
function node_filter_line($text) {
  /*
  ** If HTML mark-up is being used, strip regular mark-up:
  */

  if (strstr($text, "<br />") || strstr($text, "<p>")) {
Dries's avatar
   
Dries committed
102
    $text = ereg_replace("[\r\n]", "", $text);
Dries's avatar
   
Dries committed
103
104
105
106
107
108
109
110
111
112
113
114
  }

  /*
  ** Replace '<br>', '<br />', '<p>' and '<p />' by '\n':
  */

  $text = eregi_replace("<br>", "\n", $text);
  $text = eregi_replace("<br />", "\n", $text);
  $text = eregi_replace("<p>", "\n", $text);
  $text = eregi_replace("<br />", "\n", $text);

  /*
Dries's avatar
   
Dries committed
115
  ** Replace "\r\n" by "\n":
Dries's avatar
   
Dries committed
116
117
118
119
120
121
122
123
124
125
126
  */

  $text = ereg_replace("\r\n", "\n", $text);

  /*
  ** Replace some new line charachters:
  */

  while (strpos($text, "\n\n\n")) {
    $text = ereg_replace("\n\n\n", "\n\n", $text);
  }
Dries's avatar
   
Dries committed
127

Dries's avatar
   
Dries committed
128
129
130
  return trim($text);
}

Dries's avatar
CHANGES    
Dries committed
131
132
133
function node_filter($text) {
  if (variable_get("filter_html", 0)) $text = node_filter_html($text);
  if (variable_get("filter_link", 0)) $text = node_filter_link($text);
Dries's avatar
   
Dries committed
134
  return node_filter_line($text);
Dries's avatar
CHANGES    
Dries committed
135
136
}

137
function node_link($type, $node = 0, $main = 0) {
Dries's avatar
   
Dries committed
138

Dries's avatar
   
Dries committed
139
  if ($type == "admin" && user_access("administer nodes")) {
Dries's avatar
   
Dries committed
140
141
142
    $links[] = "<a href=\"admin.php?mod=node\">content management</a>";
  }

Dries's avatar
   
Dries committed
143
  if ($type == "page" && user_access("post content")) {
Dries's avatar
   
Dries committed
144
    $links[] = "<a href=\"module.php?mod=node&op=add\">submit</a>";
Dries's avatar
   
Dries committed
145
146
  }

Dries's avatar
   
Dries committed
147
  if ($type == "node") {
Kjartan's avatar
Kjartan committed
148
149
150
    if ($node->links) {
      $links = $node->links;
    }
Dries's avatar
   
Dries committed
151

Dries's avatar
   
Dries committed
152
    if ($main == 1 && $node->teaser != $node->body) {
Dries's avatar
   
Dries committed
153
      $links[] = "<a href=\"node.php?id=$node->nid\">". t("read more") ."</a>";
Dries's avatar
   
Dries committed
154
    }
Dries's avatar
   
Dries committed
155
156

    if (user_access("administer nodes")) {
Dries's avatar
   
Dries committed
157
       $links[] = "<a href=\"admin.php?mod=node&op=edit&id=$node->nid\">". t("administer") ."</a>";
Dries's avatar
   
Dries committed
158
    }
Dries's avatar
   
Dries committed
159
160
161
162
163
  }

  return $links ? $links : array();
}

Dries's avatar
   
Dries committed
164
function node_admin_settings($edit = array()) {
Dries's avatar
   
Dries committed
165
166
  global $op;

Dries's avatar
   
Dries committed
167
168
169
170
  if ($op == t("Save configuration")) {
    /*
    ** Save the configuration options:
    */
Dries's avatar
Dries committed
171

Dries's avatar
   
Dries committed
172
173
174
    foreach ($edit as $name => $value) {
      variable_set($name, $value);
    }
Dries's avatar
 
Dries committed
175
176
  }

Dries's avatar
   
Dries committed
177
178
179
180
  if ($op == t("Reset to defaults")) {
    /*
    ** Reset the configuration options to their default value:
    */
Dries's avatar
 
Dries committed
181

Dries's avatar
   
Dries committed
182
183
184
    foreach ($edit as $name=>$value) {
      variable_del($name);
    }
Dries's avatar
   
Dries committed
185
  }
Dries's avatar
Dries committed
186

Dries's avatar
   
Dries committed
187
  $output .= "<h3>". t("Global node settings") ."</h3>";
Dries's avatar
   
Dries committed
188
  $output .= node_conf_options();
Dries's avatar
   
Dries committed
189
190
191

  foreach (module_list() as $name) {
    if (module_hook($name, "conf_options") && module_hook($name, "node")) {
Dries's avatar
   
Dries committed
192
      $output .= "<h3>". ucfirst(module_invoke($name, "node", "name") ." settings") ."</h3>";
Dries's avatar
   
Dries committed
193
194
195
196
      $output .= module_invoke($name, "conf_options");
    }
  }

Dries's avatar
   
Dries committed
197
198
  $output .= form_submit(t("Save configuration"));
  $output .= form_submit(t("Reset to defaults"));
Dries's avatar
Dries committed
199

Dries's avatar
   
Dries committed
200
  return form($output);
Dries's avatar
   
Dries committed
201
202
}

Dries's avatar
   
Dries committed
203
function node_admin_edit($node) {
Dries's avatar
   
Dries committed
204

Dries's avatar
   
Dries committed
205
  if (is_numeric($node)) {
Dries's avatar
   
Dries committed
206
    $node = node_load(array("nid" => $node));
Dries's avatar
   
Dries committed
207
  }
Dries's avatar
   
Dries committed
208

Dries's avatar
   
Dries committed
209
210
211
  /*
  ** Edit node:
  */
Dries's avatar
   
Dries committed
212

Dries's avatar
   
Dries committed
213
  $output .= "<h3>". t("Edit") ." ". module_invoke($node->type, "node", "name") ."</h3>";
Dries's avatar
   
Dries committed
214

Dries's avatar
   
Dries committed
215
  $output .= node_form($node);
Dries's avatar
   
Dries committed
216

Dries's avatar
   
Dries committed
217
218
219
220
221
222
223
224
225
  /*
  ** Edit revisions:
  */

  if ($node->revisions) {
    $output .= "<h3>". t("Edit revisions") ."</h3>";
    $output .= "<table border=\"1\" cellpadding=\"2\" cellspacing=\"2\">";
    $output .= " <tr><th>older revisions</th><th colspan=\"3\">operations</th></tr>";
    foreach ($node->revisions as $key => $revision) {
Dries's avatar
   
Dries committed
226
      $output .= " <tr><td>". sprintf(t("revision #%d revised by %s on %s"), $key, format_name(user_load(array("uid" => $revision["uid"]))), format_date($revision["timestamp"], "small")) . ($revision["history"] ? "<br /><small>". $revision["history"] ."</small>" : "") ."</td><td><a href=\"node.php?id=$node->nid&revision=$key\">". t("view revision") ."</a></td><td><a href=\"admin.php?mod=node&op=rollback+revision&id=$node->nid&revision=$key\">". t("rollback revision") ."</a></td><td><a href=\"admin.php?mod=node&op=delete+revision&id=$node->nid&revision=$key\">". t("delete revision") ."</a></td></tr>";
Dries's avatar
   
Dries committed
227
228
229
230
    }
    $output .= "</table>";
  }

Dries's avatar
   
Dries committed
231
  /*
Dries's avatar
   
Dries committed
232
  ** Display the node form extensions:
Dries's avatar
   
Dries committed
233
  */
Dries's avatar
   
Dries committed
234

Dries's avatar
   
Dries committed
235
236
  foreach (module_list() as $name) {
    $output .= module_invoke($name, "node_link", $node);
Dries's avatar
Dries committed
237
238
  }

Dries's avatar
   
Dries committed
239
  return $output;
Dries's avatar
   
Dries committed
240
241
242

}

Dries's avatar
   
Dries committed
243
244
function node_admin_nodes() {
  global $query;
Dries's avatar
   
Dries committed
245

Dries's avatar
   
Dries committed
246
  $queries = array(array("ORDER BY n.created DESC", "new nodes"), array("ORDER BY n.changed DESC", "updated nodes"), array("WHERE n.status = 1 AND n.moderate = 0 ORDER BY n.nid DESC", "published nodes"), array("WHERE n.status = 0 AND n.moderate = 0 ORDER BY n.nid DESC", "non-published nodes"), array("WHERE n.status = 1 AND n.moderate = 1 ORDER BY n.nid DESC", "pending nodes"), array("WHERE n.status = 1 AND n.promote = 1 ORDER BY n.nid DESC", "promoted nodes"));
Dries's avatar
   
Dries committed
247

Dries's avatar
   
Dries committed
248
  $result = db_query("SELECT n.*, u.name, u.uid FROM node n LEFT JOIN users u ON n.uid = u.uid ". $queries[$query ? $query : 0][0] ." LIMIT 50");
Dries's avatar
   
Dries committed
249

Dries's avatar
   
Dries committed
250
251
  foreach ($queries as $key => $value) {
    $links[] = "<a href=\"admin.php?mod=node&op=nodes&query=$key\">$value[1]</a>";
Dries's avatar
Dries committed
252
253
  }

Dries's avatar
   
Dries committed
254
  $output .= "<small>". implode(" :: ", $links) ."</small><hr />";
Dries's avatar
   
Dries committed
255

Dries's avatar
   
Dries committed
256
257
258
  $output .= "<table border=\"1\" cellpadding=\"2\" cellspacing=\"2\">\n";
  $output .= " <tr><th>title</th><th>type</th><th>author</th><th>status</th><th colspan=\"2\">operations</th></tr>\n";
  while ($node = db_fetch_object($result)) {
Dries's avatar
   
Dries committed
259
    $output .= "<tr><td><a href=\"node.php?id=$node->nid\">". check_output($node->title) ."</a></td><td>$node->type</td><td nowrap=\"nowrap\">". format_name($node) ."</td><td>". ($node->status ? t("published") : t("not published")) ."</td><td nowrap=\"nowrap\"><a href=\"admin.php?mod=node&op=edit&id=$node->nid\">". t("edit node") ."</a></td><td nowrap=\"nowrap\"><a href=\"admin.php?mod=node&op=delete&id=$node->nid\">". t("delete node") ."</a></td></tr>";
Dries's avatar
   
Dries committed
260
  }
Dries's avatar
   
Dries committed
261
  $output .= "</table>";
Dries's avatar
   
Dries committed
262

Dries's avatar
   
Dries committed
263
  return $output;
Dries's avatar
Dries committed
264
265
}

Dries's avatar
   
Dries committed
266
267
268
269
270
271
272
273
274
275
276
277
/*
** Return the revision with the specified revision number.
*/

function node_revision_load($node, $revision) {
  return $node->revisions[$revision]["node"];
}

/*
** Create and return a new revision of the given node.
*/

Dries's avatar
   
Dries committed
278
279
280
function node_revision_create($node) {
  global $user;

Dries's avatar
   
Dries committed
281
282
283
284
285
  /*
  ** 'revision' is the name of the field used to indicicate that we
  ** have to create a new revision of a node.
  */

Dries's avatar
   
Dries committed
286
  if ($node->nid && $node->revision) {
Dries's avatar
   
Dries committed
287
288
289
290
    $prev = node_load(array("nid" => $node->nid));
    $node->revisions = $prev->revisions;
    unset($prev->revisions);
    $node->revisions[] = array("uid" => $user->uid, "timestamp" => time(), "node" => $prev, "history" => $node->history);
Dries's avatar
   
Dries committed
291
292
293
294
295
  }

  return $node;
}

Dries's avatar
   
Dries committed
296
297
298
/*
** Roll-back to the revision with the specified revision number.
*/
Dries's avatar
   
Dries committed
299

Dries's avatar
   
Dries committed
300
301
function node_revision_rollback($node, $revision) {
  global $user;
Dries's avatar
   
Dries committed
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336

  /*
  ** Extract the specified revision:
  */

  $rev = $node->revisions[$revision]["node"];

  /*
  ** Inherit all the past revisions:
  */

  $rev->revisions = $node->revisions;

  /*
  ** Save the original/current node:
  */

  $rev->revisions[] = array("uid" => $user->uid, "timestamp" => time(), "node" => $node);

  /*
  ** Remove the specified revision:
  */

  unset($rev->revisions[$revision]);

  /*
  ** Save the node:
  */

  foreach ($node as $key => $value) {
    $filter[] = $key;
  }

  node_save($rev, $filter);

Dries's avatar
   
Dries committed
337
  watchdog("special", "$node->type: rollbacked to revision #$revision of '$node->title'");
Dries's avatar
   
Dries committed
338
339
}

Dries's avatar
   
Dries committed
340
341
342
343
344
/*
** Delete the revision with specified revision number.
*/

function node_revision_delete($node, $revision) {
Dries's avatar
   
Dries committed
345
346

  unset($node->revisions[$revision]);
Dries's avatar
   
Dries committed
347

Dries's avatar
   
Dries committed
348
  node_save($node, array("nid", "revisions"));
Dries's avatar
   
Dries committed
349

Dries's avatar
   
Dries committed
350
  watchdog("special", "$node->type: removed revision #$revision of '$node->title'");
Dries's avatar
   
Dries committed
351
352
}

Dries's avatar
   
Dries committed
353
354
355
356
357
358
359
360
361
362
363
/*
** Return a list of all the existing revision numbers.
*/

function node_revision_list($node) {
  if (is_array($node->revisions)) {
    return array_keys($node->revisions);
  }
  else {
    return array();
  }
Dries's avatar
   
Dries committed
364
365
}

Dries's avatar
 
Dries committed
366
function node_admin() {
Dries's avatar
   
Dries committed
367
  global $op, $id, $revision, $edit;
Dries's avatar
   
Dries committed
368

Dries's avatar
   
Dries committed
369
  if (user_access("administer nodes")) {
Dries's avatar
   
Dries committed
370

Dries's avatar
   
Dries committed
371
372
373
    /*
    ** Compile a list of the administrative links:
    */
Dries's avatar
Dries committed
374

Dries's avatar
   
Dries committed
375
376
377
378
    $links[] = "<a href=\"admin.php?mod=node&op=nodes\">nodes</a>";
    $links[] = "<a href=\"admin.php?mod=node&op=search\">search content</a>";
    $links[] = "<a href=\"admin.php?mod=node&op=settings\">settings</a>";
    $links[] = "<a href=\"admin.php?mod=node&op=help\">help</a>";
Dries's avatar
   
Dries committed
379

Dries's avatar
   
Dries committed
380
    print "<small>". implode(" &middot; ", $links) ."</small><hr />";
Dries's avatar
   
Dries committed
381
382
383
384
385
386

    switch ($op) {
      case "help":
        print node_help();
        break;
      case "search":
Dries's avatar
   
Dries committed
387
        print search_type("node", "admin.php?mod=node&op=search");
Dries's avatar
   
Dries committed
388
        break;
Dries's avatar
   
Dries committed
389
390
391
392
      case t("Save configuration"):
      case t("Reset to defaults"):
      case "settings":
        print node_admin_settings($edit);
Dries's avatar
   
Dries committed
393
394
        break;
      case "edit":
Dries's avatar
   
Dries committed
395
        print node_admin_edit($id);
Dries's avatar
   
Dries committed
396
        break;
Dries's avatar
   
Dries committed
397
398
399
      case "delete":
        print node_delete(array("nid" => $id));
        break;
Dries's avatar
   
Dries committed
400
      case "rollback revision":
Dries's avatar
   
Dries committed
401
402
        print node_revision_rollback(node_load(array("nid" => $id)), $revision);
        print node_admin_edit($id);
Dries's avatar
   
Dries committed
403
404
        break;
      case "delete revision":
Dries's avatar
   
Dries committed
405
406
        print node_revision_delete(node_load(array("nid" => $id)), $revision);
        print node_admin_edit($id);
Dries's avatar
   
Dries committed
407
        break;
Dries's avatar
   
Dries committed
408
      case t("Preview"):
Dries's avatar
   
Dries committed
409
        print node_preview($edit);
Dries's avatar
   
Dries committed
410
        break;
Dries's avatar
   
Dries committed
411
      case t("Submit"):
Dries's avatar
   
Dries committed
412
413
414
        print node_submit($edit);
        break;
      case t("Delete"):
Dries's avatar
   
Dries committed
415
        print node_delete($edit);
Dries's avatar
   
Dries committed
416
        break;
Dries's avatar
   
Dries committed
417
      default:
Dries's avatar
   
Dries committed
418
        print node_admin_nodes();
Dries's avatar
   
Dries committed
419
420
421
422
    }
  }
  else {
    print message_access();
Dries's avatar
 
Dries committed
423
424
425
  }
}

Dries's avatar
   
Dries committed
426
427
428
429
function node_block() {
  global $theme;

  $block[0][subject] = t("Syndicate");
Dries's avatar
   
Dries committed
430
  $block[0][content] = "<div align=\"center\"><a href=\"module.php?mod=node&op=feed\"><img src=\"". $theme->image("xml.gif") ."\" width=\"36\" height=\"14\" border=\"0\" alt=\"XML\" /></a></div>\n";
Dries's avatar
   
Dries committed
431
432
433
434
435
436
  $block[0][info] = "Syndicate";

  return $block;
}

function node_feed() {
Dries's avatar
   
Dries committed
437

Dries's avatar
   
Dries committed
438
  $result = db_query("SELECT nid, type FROM node WHERE promote = '1' AND status = '1' ORDER BY created DESC LIMIT 15");
Dries's avatar
   
Dries committed
439
440

  while ($node = db_fetch_object($result)) {
Dries's avatar
   
Dries committed
441
    $item = node_load(array("nid" => $node->nid, "type" => $node->type));
Dries's avatar
   
Dries committed
442
443
444

    $link = path_uri() ."node.php?id=$item->nid";

Dries's avatar
   
Dries committed
445
    $items .= format_rss_item($item->title, $link, $item->teaser);
Dries's avatar
   
Dries committed
446
447
448
  }

  $output .= "<?xml version=\"1.0\" encoding=\"ISO-8859-1\" ?>\n";
Dries's avatar
   
Dries committed
449
  // $output .= "<!DOCTYPE rss [<!ENTITY % HTMLlat1 PUBLIC \"-//W3C//ENTITIES Latin 1 for XHTML//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml-lat1.ent\">\n";
Dries's avatar
   
Dries committed
450
  $output .= "<rss version=\"0.91\">\n";
Dries's avatar
   
Dries committed
451
  $output .= format_rss_channel(variable_get("site_name", "drupal"), path_uri(), variable_get("site_slogan", ""), $items);
Dries's avatar
   
Dries committed
452
453
  $output .= "</rss>\n";

Dries's avatar
   
Dries committed
454
455
  header("Content-Type: text/xml");

Dries's avatar
   
Dries committed
456
457
458
459
  print $output;

}

Dries's avatar
   
Dries committed
460

461
function node_validate($node, &$error) {
Dries's avatar
   
Dries committed
462
463
464
465
466
467
468

  global $user;

  /*
  ** Convert the node to an object if necessary:
  */

Dries's avatar
   
Dries committed
469
  $node = node_object($node);
Dries's avatar
   
Dries committed
470

Dries's avatar
   
Dries committed
471

Dries's avatar
   
Dries committed
472
473
474
475
  /*
  ** Validate the title field:
  */

Dries's avatar
   
Dries committed
476
  if (isset($node->title) && !$node->title) {
Dries's avatar
   
Dries committed
477
478
479
480
481
482
483
484
485
    $error["title"] = "<div style=\"color: red;\">". t("You have to specify a valid title.") ."</div>";
  }

  if (user_access("administer nodes")) {

    /*
    ** Setup default values if required:
    */

Dries's avatar
   
Dries committed
486
487
    if (!$node->created) {
      $node->created = time();
Dries's avatar
   
Dries committed
488
489
    }

Dries's avatar
   
Dries committed
490
491
    if (!$node->date) {
      $node->date = date("M j, Y g:i a", $node->created);
Dries's avatar
   
Dries committed
492
493
494
495
496
497
    }

    /*
    ** Validate the "authored by"-field:
    */

Dries's avatar
   
Dries committed
498
499
500
501
502
503
504
505
506
    if (empty($node->name)) {
      /*
      ** The use of empty() is mandatory in the context of usernames
      ** as the empty string denotes the anonymous user.  In case we
      ** are dealing with an anomymous user we set the user ID to 0.
      */
      $node->uid = 0;
    }
    else if ($account = user_load(array("name" => $node->name))) {
Dries's avatar
   
Dries committed
507
      $node->uid = $account->uid;
Dries's avatar
   
Dries committed
508
509
    }
    else {
Dries's avatar
   
Dries committed
510
      $error["name"] = "<div style=\"color: red;\">". sprintf(t("The name '%s' does not exist."), $node->name) ."</div>";
Dries's avatar
   
Dries committed
511
512
513
514
515
516
    }

    /*
    ** Validate the "authored on"-field:
    */

Dries's avatar
   
Dries committed
517
518
    if (strtotime($node->date) > 1000) {
      $node->created = strtotime($node->date);
Dries's avatar
   
Dries committed
519
520
521
522
    }
    else {
      $error["date"] = "<div style=\"color: red;\">". t("You have to specifiy a valid date.") ."</div>";
    }
Dries's avatar
   
Dries committed
523

Dries's avatar
   
Dries committed
524
525
  }

Dries's avatar
   
Dries committed
526
  return $node;
Dries's avatar
   
Dries committed
527
528
}

Dries's avatar
   
Dries committed
529

Dries's avatar
   
Dries committed
530
531
function node_form($edit) {

Dries's avatar
   
Dries committed
532
533
534
535
536
537
538
  /*
  ** Save the referer.  We record where the user came from such that we
  ** can redirect him after having completed the node forms.
  */

  referer_save();

Dries's avatar
   
Dries committed
539
540
541
542
  /*
  ** Validate the node:
  */

543
  $edit = node_validate($edit, $error);
Dries's avatar
   
Dries committed
544

Dries's avatar
   
Dries committed
545
546
547
548
549
550
551
552
  /*
  ** Generate a teaser when necessary:
  */

  if ($edit->body && !$edit->teaser) {
    $edit->teaser = node_teaser($edit->body);
  }

Dries's avatar
   
Dries committed
553
554
555
556
557
558
  /*
  ** Get the node specific bits:
  */

  $function = $edit->type ."_form";
  if (function_exists($function)) {
559
    $form .= $function($edit, $help, $error);
Dries's avatar
   
Dries committed
560
561
562
563
564
565
566
567
568
569
  }

  /*
  ** Add the help text:
  */

  if ($help) {
    $output .= "<p>$help</p>";
  }

Dries's avatar
   
Dries committed
570
571
572
573
  $output .= "<table border=\"0\" cellpadding=\"2\" cellspacing=\"2\">";
  $output .= " <tr>";
  $output .= "  <td valign=\"top\">";

Dries's avatar
   
Dries committed
574
575
576
577
578
579
580
  /*
  ** Add the default fields:
  */

  $output .= form_textfield(t("Title"), "title", $edit->title, 60, 64, $error["title"]);

  /*
Dries's avatar
   
Dries committed
581
  ** Add the node specific fields:
Dries's avatar
   
Dries committed
582
583
  */

Dries's avatar
   
Dries committed
584
  $output .= $form;
Dries's avatar
   
Dries committed
585
586
587
588
589
590
591
592
593

  /*
  ** Add the hidden fields:
  */

  if ($edit->nid) {
    $output .= form_hidden("nid", $edit->nid);
  }

Dries's avatar
   
Dries committed
594
595
596
597
598
  if (isset($edit->uid)) {
      /*
      ** The use of isset() is mandatory in the context of user IDs as uid
      ** 0 denotes the anonymous user.
      */
Dries's avatar
   
Dries committed
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
    $output .= form_hidden("uid", $edit->uid);
  }

  if ($edit->created) {
    $output .= form_hidden("created", $edit->created);
  }

  $output .= form_hidden("type", $edit->type);

  /*
  ** Add the buttons:
  */

  $output .= form_submit(t("Preview"));

  if ($edit->title && $edit->type && !$error) {
    $output .= form_submit(t("Submit"));
  }

Dries's avatar
   
Dries committed
618
  if ($edit->nid && node_access("delete", $edit)) {
Dries's avatar
   
Dries committed
619
620
621
622
623
624
625
626
    $output .= form_submit(t("Delete"));
  }

  /*
  ** Add the admin specific parts:
  */

  if (user_access("administer nodes")) {
Dries's avatar
   
Dries committed
627
    $output .= "</td><td align=\"left\" valign=\"top\">";
Dries's avatar
   
Dries committed
628

Dries's avatar
   
Dries committed
629
630
631
    $output .= form_textfield(t("Authored by"), "name", $edit->name, 20, 25, $error["name"]);
    $output .= form_textfield(t("Authored on"), "date", $edit->date, 20, 25, $error["date"]);
    $output .= "<br />";
Dries's avatar
   
Dries committed
632
633
634
635
    $output .= form_select(t("Set public/published"), "status", $edit->status, array("Disabled", "Enabled"));
    $output .= form_select(t("Allow users comments"), "comment", $edit->comment, array("Disabled", "Enabled"));
    $output .= form_select(t("Promote to front page"), "promote", $edit->promote, array("Disabled", "Enabled"));
    $output .= form_select(t("Create new revision"), "revision", $edit->revision, array("Disabled", "Enabled"));
Dries's avatar
   
Dries committed
636
637
  }

Dries's avatar
   
Dries committed
638
639
640
641
  $output .= "  </td>";
  $output .= " </tr>";
  $output .= "</table>";

Dries's avatar
   
Dries committed
642
643
644
645
646
647
  return form($output);
}

function node_add($type) {
  global $user;

Dries's avatar
   
Dries committed
648
  /*
Dries's avatar
   
Dries committed
649
  ** If a node type has been specified, validate it existence.  If no
Dries's avatar
   
Dries committed
650
651
652
  ** (valid) node type has been provied, display a node type overview.
  */

Dries's avatar
   
Dries committed
653
  if ($type && node_access("create", $type)) {
Dries's avatar
   
Dries committed
654
    $output = node_form(array("uid" => $user->uid, "name" => $user->name, "type" => $type));
Dries's avatar
   
Dries committed
655
656
  }
  else {
Dries's avatar
   
Dries committed
657

Dries's avatar
   
Dries committed
658
659
660
    /*
    ** Compile a list with the different node types and their explanation:
    */
Dries's avatar
   
Dries committed
661

Dries's avatar
   
Dries committed
662
    foreach (module_list() as $name) {
Dries's avatar
   
Dries committed
663
664
665
666
667
      if (module_hook($name, "node") && node_access("create", array("type" => $name))) {
        $output .= "<li>";
        $output .= " <a href=\"module.php?mod=node&op=add&type=$name\">". module_invoke($name, "node", "name") ."</a>";
        $output .= " <div style=\"margin-left: 20px;\">". module_invoke($name, "node", "description") ."</div>";
        $output .= "</li>";
Dries's avatar
   
Dries committed
668
      }
Dries's avatar
   
Dries committed
669
670
    }

Dries's avatar
   
Dries committed
671
672
    $output = t("Choose the appropriate item from the list:") ."<ul>$output</ul>";

Dries's avatar
   
Dries committed
673
674
675
676
677
678
679
680
681
682
  }

  return $output;
}

function node_edit($id) {
  global $user;

  $node = node_load(array("nid" => $id));

Dries's avatar
   
Dries committed
683
684
685
686
687
688
689
690
  if (node_access("update", $node)) {
    $output = node_form($node);
  }
  else {
    $output = message_access();
  }

  return $output;
Dries's avatar
   
Dries committed
691
692
693
694
695
696
697
698
}

function node_preview($edit) {

  /*
  ** Load the user's name when needed:
  */

Dries's avatar
   
Dries committed
699
700
701
702
703
704
705
706
707
708
709
  if (isset($edit["name"])) {
      /*
      ** The use of isset() is mandatory in the context of user IDs as uid
      ** 0 denotes the anonymous user.
      */
    if ($user = user_load(array("name" => $edit["name"]))) {
      $edit["uid"] = $user->uid;
    }
    else {
      $edit["uid"] = 0; // anonymous user
    }
Dries's avatar
   
Dries committed
710
711
712
713
714
715
716
717
718
719
720
721
722
723
  }
  else if ($edit["uid"]) {
    $user = user_load(array("uid" => $edit["uid"]));
    $edit["name"] = $user->name;
  }

  /*
  ** Set the created time when needed:
  */

  if (empty($edit["nid"])) {
    $edit["created"] = time();
  }

Dries's avatar
   
Dries committed
724
725
726
727
728
  /*
  ** Apply the required filters:
  */

  foreach ($edit as $key => $value) {
Dries's avatar
   
Dries committed
729
    $node->$key = check_output($value);
Dries's avatar
   
Dries committed
730
731
      /*
      ** NOTE: we can't do a check_query() or check_input() here as they
Dries's avatar
   
Dries committed
732
      **       add slashes which results in breakage.
Dries's avatar
   
Dries committed
733
      */
Dries's avatar
   
Dries committed
734
735
  }

Dries's avatar
   
Dries committed
736
737
738
739
  /*
  ** Display a preview of the node:
  */

Dries's avatar
   
Dries committed
740
  node_view($node);
Dries's avatar
   
Dries committed
741
742
743
744

  return node_form($edit);
}

Dries's avatar
   
Dries committed
745
function node_submit($node) {
Dries's avatar
   
Dries committed
746
  global $theme, $user;
Dries's avatar
   
Dries committed
747

Dries's avatar
   
Dries committed
748
  if (user_access("post content")) {
Dries's avatar
   
Dries committed
749

Dries's avatar
   
Dries committed
750
751
752
753
    /*
    ** Verify a user's submission rate and avoid duplicate nodes being
    ** inserted:
    */
Dries's avatar
   
Dries committed
754

Dries's avatar
   
Dries committed
755
    throttle("node", variable_get("max_node_rate", 900));
Dries's avatar
   
Dries committed
756

Dries's avatar
   
Dries committed
757
758
759
    /*
    ** Fixup the node when required:
    */
Dries's avatar
   
Dries committed
760

761
    $node = node_validate($node, $error);
Dries's avatar
   
Dries committed
762

Dries's avatar
   
Dries committed
763
764
765
    /*
    ** Apply the filters:
    */
Dries's avatar
   
Dries committed
766

Dries's avatar
   
Dries committed
767
768
769
    $node->teaser = filter($node->teaser);
    $node->title = filter($node->title);
    $node->body = filter($node->body);
Dries's avatar
   
Dries committed
770
771

    /*
Dries's avatar
   
Dries committed
772
    ** Create a new revision when required:
Dries's avatar
   
Dries committed
773
774
    */

Dries's avatar
   
Dries committed
775
776
777
    $node = node_revision_create($node);

    if ($node->nid) {
Dries's avatar
   
Dries committed
778
779

      /*
Dries's avatar
   
Dries committed
780
781
      ** Check whether the current user has the proper access rights to
      ** perform this operation:
Dries's avatar
   
Dries committed
782
783
      */

Dries's avatar
   
Dries committed
784
785
786
787
788
789
790
791
792
793
794
      if (node_access("update", $node)) {

        /*
        ** Compile a list of the node fields and their default values that users
        ** and administrators are allowed to save when updating a node.
        */

        if (user_access("administer nodes")) {
          $fields = array("nid", "uid", "body", "comment", "created", "promote", "moderate", "revisions", "status", "teaser", "title", "type" => $node->type);
        }
        else {
Dries's avatar
   
Dries committed
795
          $fields = array("nid", "uid" => ($user->uid ? $user->uid : 0), "body", "teaser", "title", "type" => $node->type);
Dries's avatar
   
Dries committed
796
797
        }

Dries's avatar
   
Dries committed
798
        $nid = node_save($node, array_merge($fields, module_invoke($node->type, "save", "update", $node)));
Dries's avatar
   
Dries committed
799
800
801

        watchdog("special", "$node->type: updated '$node->title'");
        $output = t("The node has been updated.");
Dries's avatar
   
Dries committed
802
803
      }
      else {
Dries's avatar
   
Dries committed
804
805
        watchdog("warning", "$node->type: not authorized to update node");
        $output = t("You are not authorized to update this node.");
Dries's avatar
   
Dries committed
806
807
      }

Dries's avatar
   
Dries committed
808
809
810
    }
    else {

Dries's avatar
   
Dries committed
811
812
813
814
      /*
      ** Check whether the current user has the proper access rights to
      ** perform this operation:
      */
Dries's avatar
   
Dries committed
815

Dries's avatar
   
Dries committed
816
      if (node_access("create", $node)) {
Dries's avatar
   
Dries committed
817

Dries's avatar
   
Dries committed
818
819
820
821
        /*
        ** Compile a list of the node fields and their default values that users
        ** and administrators are allowed to save when inserting a new node.
        */
Dries's avatar
   
Dries committed
822

Dries's avatar
   
Dries committed
823
824
825
826
        if (user_access("administer nodes")) {
          $fields = array("uid", "body", "comment" => 1, "promote", "moderate", "status" => 1, "teaser", "title", "type" => $node->type);
        }
        else {
Dries's avatar
   
Dries committed
827
          $fields = array("uid" => ($user->uid ? $user->uid : 0), "body", "comment" => 1, "teaser", "title", "type" => $node->type);
Dries's avatar
   
Dries committed
828
829
        }

Dries's avatar
   
Dries committed
830
        $nid = node_save($node, array_merge($fields, module_invoke($node->type, "save", "create", $node)));
Dries's avatar
   
Dries committed
831

Dries's avatar
   
Dries committed
832
833
        watchdog("special", "$node->type: added '$node->title'");
        $output = t("Thanks for your submission.");
Dries's avatar
   
Dries committed
834
835
      }
      else {
Dries's avatar
   
Dries committed
836
837
        watchdog("warning", "$node->type: not authorized to create node");
        $output = t("You are not authorized to create this node.");
Dries's avatar
   
Dries committed
838
839
      }
    }
Dries's avatar
   
Dries committed
840

Dries's avatar
   
Dries committed
841
842
843
844
845
846
847
848
849
850
851
    /*
    ** Reload the node from the database:
    */

    $node = node_load(array("nid" => $nid));

    /*
    ** For usability's sake, make sure to present the user with some
    ** useful links as where to go next.
    */

Dries's avatar
   
Dries committed
852
    if ($referer = referer_load()) {
Dries's avatar
   
Dries committed
853
854
855
856
857
858
859
860
861
      $links[] = "<a href=\"$referer\">". t("return") ."</a>";
    }

    if ($nid && node_access("view", $node)) {
      $links[] = "<a href=\"node.php?id=$nid\">". t("view") ."</a>";
    }

    if ($nid && node_access("update", $node)) {
      $links[] = "<a href=\"module.php?mod=node&op=edit&id=$nid\">". t("edit") ."</a>";
Dries's avatar
   
Dries committed
862
    }
Dries's avatar
   
Dries committed
863
864

    $output .= "<p>". $theme->links($links) ."</p>";
Dries's avatar
   
Dries committed
865
  }
Dries's avatar
   
Dries committed
866
867
868
  else {
    $output = message_access();
  }
Dries's avatar
   
Dries committed
869
870
871
872

  return $output;
}

Dries's avatar
   
Dries committed
873
function node_delete($edit) {
Dries's avatar
   
Dries committed
874

Dries's avatar
   
Dries committed
875
876
877
  $node = node_load(array("nid" => $edit["nid"]));

  if (node_access("delete", $node)) {
Dries's avatar
   
Dries committed
878

Dries's avatar
   
Dries committed
879
    if ($edit["confirm"]) {
Dries's avatar
   
Dries committed
880

Dries's avatar
   
Dries committed
881
882
883
884
885
886
887
888
889
890
891
      /*
      ** Delete the specified node and its comments:
      */

      db_query("DELETE FROM node WHERE nid = '$node->nid'");
      db_query("DELETE FROM comments WHERE lid = '$node->nid'");

      /*
      ** Call the node specific callback (if any):
      */

892
      module_invoke($node->type, "delete", $node);
Dries's avatar
   
Dries committed
893

Dries's avatar
   
Dries committed
894
      watchdog("special", "$node->type: deleted '$node->title'");
Dries's avatar
   
Dries committed
895
896
897
      $output = t("The node has been deleted.");
    }
    else {
Dries's avatar
   
Dries committed
898
      $output .= form_item(t("Confirm deletion"), check_output($node->title));
Dries's avatar
   
Dries committed
899
900
901
      $output .= form_hidden("nid", $node->nid);
      $output .= form_hidden("confirm", 1);
      $output .= form_submit(t("Delete"));
Dries's avatar
   
Dries committed
902
      $output = form($output);
Dries's avatar
   
Dries committed
903
    }
Dries's avatar
   
Dries committed
904
905
  }
  else {
Dries's avatar
   
Dries committed
906
    watchdog("warning", "$node->type: not authorized to remove node");
Dries's avatar
   
Dries committed
907
    $output = t("You are not authorized to remove this node.");
Dries's avatar
   
Dries committed
908
909
910
911
912
  }

  return $output;
}

Dries's avatar
   
Dries committed
913
function node_page() {
Dries's avatar
   
Dries committed
914
  global $op, $id, $user, $edit, $type, $theme, $meta, $date;
Dries's avatar
   
Dries committed
915
916
917

  if ($op == "feed") {
    node_feed();
Dries's avatar
   
Dries committed
918
    return;
Dries's avatar
   
Dries committed
919
  }
Dries's avatar
   
Dries committed
920

Dries's avatar
   
Dries committed
921
922
923
924
925
926
927
928
929
930
931
932
933
934
  /*
  ** Try to find a good title:
  */

  if ($type) {
    $title = ucfirst(module_invoke($type, "node", "name"));
  }
  else if ($edit["type"]) {
    $title = ucfirst(module_invoke($edit["type"], "node", "name"));
  }
  else {
    $title = t("Submission form");
  }

Dries's avatar
   
Dries committed
935
936
937
938
  $theme->header();

  switch ($op) {
    case "add":
Dries's avatar
   
Dries committed
939
      $theme->box($title, node_add($type));
Dries's avatar
   
Dries committed
940
941
      break;
    case "edit":
Dries's avatar
   
Dries committed
942
      $theme->box($title, node_edit($id));
Dries's avatar
   
Dries committed
943
944
      break;
    case t("Preview"):
Dries's avatar
   
Dries committed
945
      $theme->box($title, node_preview($edit));
Dries's avatar
   
Dries committed
946
947
      break;
    case t("Submit"):
Dries's avatar
   
Dries committed
948
      $theme->box($title, node_submit($edit));
Dries's avatar
   
Dries committed
949
      break;
Dries's avatar
   
Dries committed
950
    case t("Delete"):
Dries's avatar
   
Dries committed
951
      $theme->box($title, node_delete($edit));
Dries's avatar
   
Dries committed
952
      break;
Dries's avatar
   
Dries committed
953
    default:
Dries's avatar
   
Dries committed
954
      $result = db_query("SELECT nid, type FROM node WHERE ". ($meta ? "attributes LIKE '%". check_input($meta) ."%' AND " : "") ." promote = '1' AND status = '1' AND created <= '". ($date > 0 ? check_input($date) : time()) ."' ORDER BY created DESC LIMIT ". ($user->nodes ? $user->nodes : variable_get("default_nodes_main", 10)));
955
      while ($node = db_fetch_object($result)) {
Dries's avatar
   
Dries committed
956
        node_view(node_load(array("nid" => $node->nid, "type" => $node->type)), 1);
957
958
      }
  }
Dries's avatar
   
Dries committed
959
960

  $theme->footer();
Dries's avatar
   
Dries committed
961
}
Dries's avatar
   
Dries committed
962

Dries's avatar
 
Dries committed
963
?>