node.module 30.4 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 19 20 21
// TODO: still used by themes, yet doesn't return anything at the moment
function node_index() {
}

function node_get_comments($nid) {
Dries's avatar
 
Dries committed
22
  $comment = db_fetch_object(db_query("SELECT COUNT(c.nid) AS number FROM node n LEFT JOIN comments c ON n.nid = c.nid WHERE n.nid = '$nid' GROUP BY n.nid"));
Dries's avatar
 
Dries committed
23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152
  return $comment->number ? $comment->number : 0;
}

function node_teaser($body) {

  $size = 400;

  /*
  ** If we have a short body, return the entire body:
  */

  if (strlen($body) < $size) {
    return $body;
  }

  /*
  ** If we have a long body, try not to split paragraphs:
  */

  if ($length = strpos($body, "\n", $size)) {
    return substr($body, 0, $length + 1);
  }

  /*
  ** If we have a long body, try not to split sentences:
  */

  return substr($body, 0, strpos($body, ". ", $size) + 1);

}

function node_invoke($node, $name, $arg = 0) {
  if (is_array($node)) {
    $function = $node[type] ."_$name";
  }
  else if (is_object($node)) {
    $function = $node->type ."_$name";
  }
  else if (is_string($node)) {
    $function = $node ."_$name";
  }

  if (function_exists($function)) {
    return ($arg ? $function($node, $arg) : $function($node));
  }
}

function node_load($conditions) {

  /*
  ** Turn the conditions into a query:
  */

  foreach ($conditions as $key => $value) {
    $cond[] = "n.". check_query($key) ." = '". check_query($value) ."'";
  }

  /*
  ** Retrieve the node:
  */

  $node = db_fetch_object(db_query("SELECT n.*, u.uid, u.name FROM node n LEFT JOIN users u ON u.uid = n.uid WHERE ". implode(" AND ", $cond)));

  /*
  ** Unserialize the revisions field:
  */

  if ($node->revisions) {
    $node->revisions = unserialize($node->revisions);
  }

  /*
  ** Call the node specific callback (if any) and piggy-back the
  ** results to the node or overwrite some values:
  */

  if ($extra = module_invoke($node->type, "load", $node)) {
    foreach ($extra as $key => $value) {
      $node->$key = $value;
    }
  }

  return $node;
}

function node_save($node, $filter) {

  $fields = array("nid", "uid", "type", "title", "teaser", "body", "revisions", "score", "status", "comment", "promote", "moderate", "created", "changed", "users", "votes");

  foreach ($filter as $key => $value) {
    /*
    ** Only save those fields specified by the filter.  If the filter
    ** does not specify a default value, use the value of the $node's
    ** corresponding field instead.
    */

    if (is_numeric($key)) {
      if (isset($node->$value)) {
          // The above check is mandatory.
        $edit->$value = $node->$value;
      }
    }
    else {
      if (isset($value)) {
          // The above check is mandatory.
        $edit->$key = $value;
      }
    }
  }

  $node = $edit;

  /*
  ** Serialize the revisions field:
  */

  if ($node->revisions) {
    $node->revisions = serialize($node->revisions);
  }

  /*
  ** Apply filters to some default node fields:
  */

  if (empty($node->nid)) {

    /*
    ** Insert a new node:
    */

Dries's avatar
 
Dries committed
153
    // Set some required fields:
Dries's avatar
 
Dries committed
154
    $node->created = time();
Dries's avatar
 
Dries committed
155
    $node->changed = time();
Dries's avatar
 
Dries committed
156 157
    $node->nid = db_result(db_query("SELECT MAX(nid) + 1 FROM node"));

Dries's avatar
 
Dries committed
158
    // Prepare the query:
Dries's avatar
 
Dries committed
159 160 161 162 163 164 165
    foreach ($node as $key => $value) {
      if (in_array($key, $fields)) {
        $k[] = check_query($key);
        $v[] = "'". check_query($value) ."'";
      }
    }

Dries's avatar
 
Dries committed
166
    // Insert the node into the database:
Dries's avatar
 
Dries committed
167 168
    db_query("INSERT INTO node (". implode(", ", $k) .") VALUES (". implode(", ", $v) .")");

Dries's avatar
 
Dries committed
169
    // Call the node specific callback (if any):
Dries's avatar
 
Dries committed
170 171 172 173 174 175 176 177
    module_invoke($node->type, "insert", $node);
  }
  else {

    /*
    ** Update an existing node:
    */

Dries's avatar
 
Dries committed
178
    // Set some required fields:
Dries's avatar
 
Dries committed
179 180
    $node->changed = time();

Dries's avatar
 
Dries committed
181
    // Prepare the query:
Dries's avatar
 
Dries committed
182 183 184 185 186 187
    foreach ($node as $key => $value) {
      if (in_array($key, $fields)) {
        $q[] = check_query($key) ." = '". check_query($value) ."'";
      }
    }

Dries's avatar
 
Dries committed
188
    // Update the node in the database:
Dries's avatar
 
Dries committed
189 190
    db_query("UPDATE node SET ". implode(", ", $q) ." WHERE nid = '$node->nid'");

Dries's avatar
 
Dries committed
191
    // Call the node specific callback (if any):
Dries's avatar
 
Dries committed
192 193 194 195 196 197 198 199 200 201 202 203 204 205 206
    module_invoke($node->type, "update", $node);

  }

  /*
  ** Return the node ID:
  */

  return $node->nid;

}

function node_view($node, $main = 0) {
  global $theme;

Dries's avatar
 
Dries committed
207
  $node = array2object($node);
Dries's avatar
 
Dries committed
208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226

  /*
  ** The "view" hook can be implemented to overwrite the default function
  ** to display nodes.
  */

  if (module_hook($node->type, "view")) {
    node_invoke($node, "view", $main);
  }
  else {

    /*
    ** Default behavior:
    */

    $theme->node($node, $main);
  }
}

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

Dries's avatar
 
Dries committed
229 230
  if (user_access("administer nodes")) {
    return 1;
Dries's avatar
 
Dries committed
231
  }
Dries's avatar
 
Dries committed
232
  else {
Dries's avatar
 
Dries committed
233

Dries's avatar
 
Dries committed
234 235 236
    /*
    ** Convert the node to an object if necessary:
    */
Dries's avatar
 
Dries committed
237

Dries's avatar
 
Dries committed
238
    $node = array2object($node);
Dries's avatar
 
Dries committed
239

Dries's avatar
 
Dries committed
240 241 242 243
    /*
    ** Construct a function:
    */

Dries's avatar
 
Dries committed
244 245 246 247 248 249 250 251
    if ($node->type) {
      $type = $node->type;
    }
    else {
      $type = $node;
    }

    $function = $type ."_access";
Dries's avatar
 
Dries committed
252 253 254 255 256 257 258

    if (function_exists($function)) {
      return $function($op, $node);
    }
    else {
      return 0;
    }
Dries's avatar
 
Dries committed
259 260 261
  }
}

Dries's avatar
 
Dries committed
262
function node_perm() {
Dries's avatar
 
Dries committed
263
  return array("administer nodes", "access content", "post content");
Dries's avatar
 
Dries committed
264 265
}

Dries's avatar
 
Dries committed
266
function node_search($keys) {
Dries's avatar
 
Dries committed
267
  global $PHP_SELF;
Dries's avatar
 
Dries committed
268 269 270

  $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
271
    $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
272 273 274 275 276
  }

  return $find;
}

Dries's avatar
 
Dries committed
277
function node_conf_options() {
Dries's avatar
 
Dries committed
278
  $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
279 280 281
  return $output;
}

Dries's avatar
CHANGES  
Dries committed
282
function node_conf_filters() {
Dries's avatar
 
Dries committed
283
  $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
284
  $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
285
  $output .= "<hr />";
Dries's avatar
 
Dries committed
286
  $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
287
  $output .= "<hr />";
Dries's avatar
CHANGES  
Dries committed
288 289 290 291 292 293 294 295 296 297 298
  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
299 300 301 302 303
  $src = array("/\[\[(([^\|]*?)(\|([^\|]*?))?)\]\]/e");  // [link|description]
  $dst = array(format_tag('\\2', '\\4'));                // [link|description]
  return preg_replace($src, $dst, $text);
}

Dries's avatar
 
Dries committed
304
function node_filter_line($text) {
Dries's avatar
 
Dries committed
305

Dries's avatar
 
Dries committed
306
  /*
Dries's avatar
 
Dries committed
307 308 309 310 311 312 313
  ** This "line break filter" will try to get the line breaks right
  ** regardless of the user's input.  Its goal aspires a consistent
  ** mark-up and use of line breaks and paragraphs.
  */

  /*
  ** If HTML mark-up is being used, strip regular line breaks:
Dries's avatar
 
Dries committed
314 315 316
  */

  if (strstr($text, "<br />") || strstr($text, "<p>")) {
Dries's avatar
 
Dries committed
317
    $text = ereg_replace("[\r\n]", "", $text);
Dries's avatar
 
Dries committed
318 319 320 321 322 323 324 325 326
  }

  /*
  ** 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);
Dries's avatar
 
Dries committed
327
  $text = eregi_replace("<p />", "\n", $text);
Dries's avatar
 
Dries committed
328 329

  /*
Dries's avatar
 
Dries committed
330
  ** Replace '\r\n' by '\n':
Dries's avatar
 
Dries committed
331 332 333 334 335 336 337 338 339 340 341
  */

  $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
342

Dries's avatar
 
Dries committed
343 344 345 346 347 348 349
  /*
  ** Replace some common "artifacts":
  */

  $list = "blockquote|li|ol|ul|table|th|td|tr|pre";
  $text = preg_replace(array("/\n\s*<([\/])($list)/", "/($list)>\s*\n/"), array("<$1$2", "$1>"), $text);

Dries's avatar
 
Dries committed
350 351 352
  return trim($text);
}

Dries's avatar
CHANGES  
Dries committed
353 354 355
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
356
  return node_filter_line($text);
Dries's avatar
CHANGES  
Dries committed
357 358
}

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

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

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

Dries's avatar
 
Dries committed
369
  if ($type == "node") {
Kjartan's avatar
Kjartan committed
370 371 372
    if ($node->links) {
      $links = $node->links;
    }
Dries's avatar
 
Dries committed
373

Dries's avatar
 
Dries committed
374
    if ($main == 1 && $node->teaser != $node->body) {
Dries's avatar
 
Dries committed
375
      $links[] = "<a href=\"node.php?id=$node->nid\">". t("read more") ."</a>";
Dries's avatar
 
Dries committed
376
    }
Dries's avatar
 
Dries committed
377 378

    if (user_access("administer nodes")) {
Dries's avatar
 
Dries committed
379
       $links[] = "<a href=\"admin.php?mod=node&op=edit&id=$node->nid\">". t("administer") ."</a>";
Dries's avatar
 
Dries committed
380
    }
Dries's avatar
 
Dries committed
381 382 383 384 385
  }

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

Dries's avatar
 
Dries committed
386
function node_admin_settings($edit = array()) {
Dries's avatar
 
Dries committed
387 388
  global $op;

Dries's avatar
 
Dries committed
389 390 391 392
  if ($op == t("Save configuration")) {
    /*
    ** Save the configuration options:
    */
Dries's avatar
Dries committed
393

Dries's avatar
 
Dries committed
394 395 396
    foreach ($edit as $name => $value) {
      variable_set($name, $value);
    }
Dries's avatar
 
Dries committed
397 398
  }

Dries's avatar
 
Dries committed
399 400 401 402
  if ($op == t("Reset to defaults")) {
    /*
    ** Reset the configuration options to their default value:
    */
Dries's avatar
 
Dries committed
403

Dries's avatar
 
Dries committed
404 405 406
    foreach ($edit as $name=>$value) {
      variable_del($name);
    }
Dries's avatar
 
Dries committed
407
  }
Dries's avatar
Dries committed
408

Dries's avatar
 
Dries committed
409
  $output .= "<h3>". t("Global node settings") ."</h3>";
Dries's avatar
 
Dries committed
410
  $output .= node_conf_options();
Dries's avatar
 
Dries committed
411 412 413

  foreach (module_list() as $name) {
    if (module_hook($name, "conf_options") && module_hook($name, "node")) {
Dries's avatar
 
Dries committed
414
      $output .= "<h3>". ucfirst(module_invoke($name, "node", "name") ." settings") ."</h3>";
Dries's avatar
 
Dries committed
415 416 417 418
      $output .= module_invoke($name, "conf_options");
    }
  }

Dries's avatar
 
Dries committed
419 420
  $output .= form_submit(t("Save configuration"));
  $output .= form_submit(t("Reset to defaults"));
Dries's avatar
Dries committed
421

Dries's avatar
 
Dries committed
422
  return form($output);
Dries's avatar
 
Dries committed
423 424
}

Dries's avatar
 
Dries committed
425
function node_admin_edit($node) {
Dries's avatar
 
Dries committed
426

Dries's avatar
 
Dries committed
427
  if (is_numeric($node)) {
Dries's avatar
 
Dries committed
428
    $node = node_load(array("nid" => $node));
Dries's avatar
 
Dries committed
429
  }
Dries's avatar
 
Dries committed
430

Dries's avatar
 
Dries committed
431 432 433
  /*
  ** Edit node:
  */
Dries's avatar
 
Dries committed
434

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

Dries's avatar
 
Dries committed
437
  $output .= node_form($node);
Dries's avatar
 
Dries committed
438

Dries's avatar
 
Dries committed
439 440 441 442 443 444 445 446 447
  /*
  ** 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
448
      $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
449 450 451 452
    }
    $output .= "</table>";
  }

Dries's avatar
 
Dries committed
453
  /*
Dries's avatar
 
Dries committed
454
  ** Display the node form extensions:
Dries's avatar
 
Dries committed
455
  */
Dries's avatar
 
Dries committed
456

Dries's avatar
 
Dries committed
457 458
  foreach (module_list() as $name) {
    $output .= module_invoke($name, "node_link", $node);
Dries's avatar
Dries committed
459 460
  }

Dries's avatar
 
Dries committed
461
  return $output;
Dries's avatar
 
Dries committed
462 463 464

}

Dries's avatar
 
Dries committed
465 466
function node_admin_nodes() {
  global $query;
Dries's avatar
 
Dries committed
467

Dries's avatar
 
Dries committed
468
  $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
469

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

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

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

Dries's avatar
 
Dries committed
478 479 480
  $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
481
    $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
482
  }
Dries's avatar
 
Dries committed
483
  $output .= "</table>";
Dries's avatar
 
Dries committed
484

Dries's avatar
 
Dries committed
485
  return $output;
Dries's avatar
Dries committed
486 487
}

Dries's avatar
 
Dries committed
488 489 490 491 492 493 494 495 496 497 498 499
/*
** 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
500 501 502
function node_revision_create($node) {
  global $user;

Dries's avatar
 
Dries committed
503 504 505 506 507
  /*
  ** '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
508
  if ($node->nid && $node->revision) {
Dries's avatar
 
Dries committed
509 510 511 512
    $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
513 514 515 516 517
  }

  return $node;
}

Dries's avatar
 
Dries committed
518 519 520
/*
** Roll-back to the revision with the specified revision number.
*/
Dries's avatar
 
Dries committed
521

Dries's avatar
 
Dries committed
522 523
function node_revision_rollback($node, $revision) {
  global $user;
Dries's avatar
 
Dries committed
524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558

  /*
  ** 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
559
  watchdog("special", "$node->type: rollbacked to revision #$revision of '$node->title'");
Dries's avatar
 
Dries committed
560 561
}

Dries's avatar
 
Dries committed
562 563 564 565 566
/*
** Delete the revision with specified revision number.
*/

function node_revision_delete($node, $revision) {
Dries's avatar
 
Dries committed
567 568

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

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

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

Dries's avatar
 
Dries committed
575 576 577 578 579 580 581 582 583 584 585
/*
** 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
586 587
}

Dries's avatar
 
Dries committed
588
function node_admin() {
Dries's avatar
 
Dries committed
589
  global $op, $id, $revision, $edit;
Dries's avatar
 
Dries committed
590

Dries's avatar
 
Dries committed
591
  if (user_access("administer nodes")) {
Dries's avatar
 
Dries committed
592

Dries's avatar
 
Dries committed
593 594 595
    /*
    ** Compile a list of the administrative links:
    */
Dries's avatar
Dries committed
596

Dries's avatar
 
Dries committed
597 598 599 600
    $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
601

Dries's avatar
 
Dries committed
602
    print "<small>". implode(" &middot; ", $links) ."</small><hr />";
Dries's avatar
 
Dries committed
603 604 605 606 607 608

    switch ($op) {
      case "help":
        print node_help();
        break;
      case "search":
Dries's avatar
 
Dries committed
609
        print search_type("node", "admin.php?mod=node&op=search");
Dries's avatar
 
Dries committed
610
        break;
Dries's avatar
 
Dries committed
611 612 613 614
      case t("Save configuration"):
      case t("Reset to defaults"):
      case "settings":
        print node_admin_settings($edit);
Dries's avatar
 
Dries committed
615 616
        break;
      case "edit":
Dries's avatar
 
Dries committed
617
        print node_admin_edit($id);
Dries's avatar
 
Dries committed
618
        break;
Dries's avatar
 
Dries committed
619 620 621
      case "delete":
        print node_delete(array("nid" => $id));
        break;
Dries's avatar
 
Dries committed
622
      case "rollback revision":
Dries's avatar
 
Dries committed
623 624
        print node_revision_rollback(node_load(array("nid" => $id)), $revision);
        print node_admin_edit($id);
Dries's avatar
 
Dries committed
625 626
        break;
      case "delete revision":
Dries's avatar
 
Dries committed
627 628
        print node_revision_delete(node_load(array("nid" => $id)), $revision);
        print node_admin_edit($id);
Dries's avatar
 
Dries committed
629
        break;
Dries's avatar
 
Dries committed
630
      case t("Preview"):
Dries's avatar
 
Dries committed
631
        print node_preview($edit);
Dries's avatar
 
Dries committed
632
        break;
Dries's avatar
 
Dries committed
633
      case t("Submit"):
Dries's avatar
 
Dries committed
634 635 636
        print node_submit($edit);
        break;
      case t("Delete"):
Dries's avatar
 
Dries committed
637
        print node_delete($edit);
Dries's avatar
 
Dries committed
638
        break;
Dries's avatar
 
Dries committed
639
      default:
Dries's avatar
 
Dries committed
640
        print node_admin_nodes();
Dries's avatar
 
Dries committed
641 642 643 644
    }
  }
  else {
    print message_access();
Dries's avatar
 
Dries committed
645 646 647
  }
}

Dries's avatar
 
Dries committed
648 649 650 651
function node_block() {
  global $theme;

  $block[0][subject] = t("Syndicate");
Dries's avatar
 
Dries committed
652
  $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
653 654 655 656 657 658
  $block[0][info] = "Syndicate";

  return $block;
}

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

Dries's avatar
 
Dries committed
660
  $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
661 662

  while ($node = db_fetch_object($result)) {
Dries's avatar
 
Dries committed
663
    $item = node_load(array("nid" => $node->nid, "type" => $node->type));
Dries's avatar
 
Dries committed
664 665 666

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

Dries's avatar
 
Dries committed
667
    $items .= format_rss_item($item->title, $link, $item->teaser);
Dries's avatar
 
Dries committed
668 669 670
  }

  $output .= "<?xml version=\"1.0\" encoding=\"ISO-8859-1\" ?>\n";
Dries's avatar
 
Dries committed
671
  // $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
672
  $output .= "<rss version=\"0.91\">\n";
Dries's avatar
 
Dries committed
673
  $output .= format_rss_channel(variable_get("site_name", "drupal"), path_uri(), variable_get("site_slogan", ""), $items);
Dries's avatar
 
Dries committed
674 675
  $output .= "</rss>\n";

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

Dries's avatar
 
Dries committed
678 679 680 681
  print $output;

}

Dries's avatar
 
Dries committed
682

683
function node_validate($node, &$error) {
Dries's avatar
 
Dries committed
684 685 686 687 688 689 690

  global $user;

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

Dries's avatar
 
Dries committed
691
  $node = array2object($node);
Dries's avatar
 
Dries committed
692 693 694 695 696

  /*
  ** Validate the title field:
  */

Dries's avatar
 
Dries committed
697
  if (isset($node->title) && !$node->title) {
Dries's avatar
 
Dries committed
698 699 700 701 702 703 704 705 706
    $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
707 708
    if (!$node->created) {
      $node->created = time();
Dries's avatar
 
Dries committed
709 710
    }

Dries's avatar
 
Dries committed
711 712
    if (!$node->date) {
      $node->date = date("M j, Y g:i a", $node->created);
Dries's avatar
 
Dries committed
713 714 715 716 717 718
    }

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

Dries's avatar
 
Dries committed
719 720 721 722 723 724 725 726 727
    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
728
      $node->uid = $account->uid;
Dries's avatar
 
Dries committed
729 730
    }
    else {
Dries's avatar
 
Dries committed
731
      $error["name"] = "<div style=\"color: red;\">". sprintf(t("The name '%s' does not exist."), $node->name) ."</div>";
Dries's avatar
 
Dries committed
732 733 734 735 736 737
    }

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

Dries's avatar
 
Dries committed
738 739
    if (strtotime($node->date) > 1000) {
      $node->created = strtotime($node->date);
Dries's avatar
 
Dries committed
740 741 742 743
    }
    else {
      $error["date"] = "<div style=\"color: red;\">". t("You have to specifiy a valid date.") ."</div>";
    }
Dries's avatar
 
Dries committed
744

Dries's avatar
 
Dries committed
745 746
  }

Dries's avatar
 
Dries committed
747
  return $node;
Dries's avatar
 
Dries committed
748 749
}

Dries's avatar
 
Dries committed
750

Dries's avatar
 
Dries committed
751 752
function node_form($edit) {

Dries's avatar
 
Dries committed
753 754 755 756 757 758 759
  /*
  ** 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
760 761 762 763
  /*
  ** Validate the node:
  */

764
  $edit = node_validate($edit, $error);
Dries's avatar
 
Dries committed
765

Dries's avatar
 
Dries committed
766 767 768 769 770 771 772 773
  /*
  ** Generate a teaser when necessary:
  */

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

Dries's avatar
 
Dries committed
774 775 776 777 778 779
  /*
  ** Get the node specific bits:
  */

  $function = $edit->type ."_form";
  if (function_exists($function)) {
780
    $form .= $function($edit, $help, $error);
Dries's avatar
 
Dries committed
781 782 783 784 785 786 787 788 789 790
  }

  /*
  ** Add the help text:
  */

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

Dries's avatar
 
Dries committed
791 792 793 794
  $output .= "<table border=\"0\" cellpadding=\"2\" cellspacing=\"2\">";
  $output .= " <tr>";
  $output .= "  <td valign=\"top\">";

Dries's avatar
 
Dries committed
795 796 797 798 799 800 801
  /*
  ** Add the default fields:
  */

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

  /*
Dries's avatar
 
Dries committed
802
  ** Add the node specific fields:
Dries's avatar
 
Dries committed
803 804
  */

Dries's avatar
 
Dries committed
805
  $output .= $form;
Dries's avatar
 
Dries committed
806 807 808 809 810 811 812 813 814

  /*
  ** Add the hidden fields:
  */

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

Dries's avatar
 
Dries committed
815 816 817 818 819
  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
820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838
    $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
839
  if ($edit->nid && node_access("delete", $edit)) {
Dries's avatar
 
Dries committed
840 841 842 843 844 845 846 847
    $output .= form_submit(t("Delete"));
  }

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

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

Dries's avatar
 
Dries committed
850 851 852
    $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
853
    $output .= form_select(t("Set public/published"), "status", $edit->status, array("Disabled", "Enabled"));
Dries's avatar
 
Dries committed
854
    $output .= form_select(t("Queue for moderation"), "moderate", $edit->moderate, array("Disabled", "Enabled"));
Dries's avatar
 
Dries committed
855
    $output .= form_select(t("Promote to front page"), "promote", $edit->promote, array("Disabled", "Enabled"));
Dries's avatar
 
Dries committed
856
    $output .= form_select(t("Allow users comments"), "comment", $edit->comment, array("Disabled", "Enabled"));
Dries's avatar
 
Dries committed
857
    $output .= form_select(t("Create new revision"), "revision", $edit->revision, array("Disabled", "Enabled"));
Dries's avatar
 
Dries committed
858 859
  }

Dries's avatar
 
Dries committed
860 861 862 863
  $output .= "  </td>";
  $output .= " </tr>";
  $output .= "</table>";

Dries's avatar
 
Dries committed
864 865 866 867 868 869
  return form($output);
}

function node_add($type) {
  global $user;

Dries's avatar
 
Dries committed
870
  /*
Dries's avatar
 
Dries committed
871
  ** If a node type has been specified, validate it existence.  If no
Dries's avatar
 
Dries committed
872 873 874
  ** (valid) node type has been provied, display a node type overview.
  */

Dries's avatar
 
Dries committed
875
  if ($type && node_access("create", $type)) {
Dries's avatar
 
Dries committed
876
    $output = node_form(array("uid" => $user->uid, "name" => $user->name, "type" => $type));
Dries's avatar
 
Dries committed
877 878
  }
  else {
Dries's avatar
 
Dries committed
879

Dries's avatar
 
Dries committed
880 881 882
    /*
    ** Compile a list with the different node types and their explanation:
    */
Dries's avatar
 
Dries committed
883

Dries's avatar
 
Dries committed
884
    foreach (module_list() as $name) {
Dries's avatar
 
Dries committed
885 886 887 888 889
      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
890
      }
Dries's avatar
 
Dries committed
891 892
    }

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

Dries's avatar
 
Dries committed
895 896 897 898 899 900 901 902 903 904
  }

  return $output;
}

function node_edit($id) {
  global $user;

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

Dries's avatar
 
Dries committed
905 906 907 908 909 910 911 912
  if (node_access("update", $node)) {
    $output = node_form($node);
  }
  else {
    $output = message_access();
  }

  return $output;
Dries's avatar
 
Dries committed
913 914
}

Dries's avatar
 
Dries committed
915 916 917 918 919 920
function node_preview($node) {

  /*
  ** Convert the array to an object:
  */

Dries's avatar
 
Dries committed
921
  $node = array2object($node);
Dries's avatar
 
Dries committed
922 923 924 925 926

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

Dries's avatar
 
Dries committed
927 928 929 930 931 932 933 934
  if (isset($node->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" => $node->name))) {
      $node->uid = $user->uid;
Dries's avatar
 
Dries committed
935 936
    }
    else {
Dries's avatar
 
Dries committed
937
      $node->uid = 0; // anonymous user
Dries's avatar
 
Dries committed
938
    }
Dries's avatar
 
Dries committed
939
  }
Dries's avatar
 
Dries committed
940
  else if ($node->uid) {
Dries's avatar
 
Dries committed
941 942
    $user = user_load(array("uid" => $node->uid));
    $node->name = $user->name;
Dries's avatar
 
Dries committed
943 944 945 946 947 948
  }

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

Dries's avatar
 
Dries committed
949 950
  if (empty($node->nid)) {
    $node->created = time();
Dries's avatar
 
Dries committed
951 952
  }

Dries's avatar
 
Dries committed
953 954 955 956
  /*
  ** Apply the required filters:
  */

Dries's avatar
 
Dries committed
957
  if ($node->nid) {
Dries's avatar
 
Dries committed
958
    $view = array_merge($node, module_invoke($node->type, "save", "update", $node));
Dries's avatar
 
Dries committed
959 960
  }
  else {
Dries's avatar
 
Dries committed
961
    $view = array_merge($node, module_invoke($node->type, "save", "create", $node));
Dries's avatar
 
Dries committed
962 963
  }

Dries's avatar
 
Dries committed
964 965 966 967
  /*
  ** Display a preview of the node:
  */

Dries's avatar
 
Dries committed
968
  node_view($view);
Dries's avatar
 
Dries committed
969

Dries's avatar
 
Dries committed
970
  return node_form($node);
Dries's avatar
 
Dries committed
971 972
}

Dries's avatar
 
Dries committed
973
function node_submit($node) {
Dries's avatar
 
Dries committed
974
  global $theme, $user;
Dries's avatar
 
Dries committed
975

Dries's avatar
 
Dries committed
976
  if (user_access("post content")) {
Dries's avatar
 
Dries committed
977

Dries's avatar
 
Dries committed
978 979 980
    /*
    ** Fixup the node when required:
    */
Dries's avatar
 
Dries committed
981

982
    $node = node_validate($node, $error);
Dries's avatar
 
Dries committed
983

Dries's avatar
 
Dries committed
984
    /*
Dries's avatar
 
Dries committed
985
    ** Create a new revision when required:
Dries's avatar
 
Dries committed
986 987
    */

Dries's avatar
 
Dries committed
988 989 990
    $node = node_revision_create($node);

    if ($node->nid) {
Dries's avatar
 
Dries committed
991 992

      /*
Dries's avatar
 
Dries committed
993 994
      ** Check whether the current user has the proper access rights to
      ** perform this operation:
Dries's avatar
 
Dries committed
995 996
      */

Dries's avatar
 
Dries committed
997 998 999 1000 1001 1002 1003 1004 1005 1006 1007
      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
1008
          $fields = array("nid", "uid" => ($user->uid ? $user->uid : 0), "body", "teaser", "title", "type" => $node->type);
Dries's avatar
 
Dries committed
1009 1010
        }

Dries's avatar
 
Dries committed
1011
        $nid = node_save($node, array_merge($fields, module_invoke($node->type, "save", "update", $node)));
Dries's avatar
 
Dries committed
1012 1013 1014

        watchdog("special", "$node->type: updated '$node->title'");
        $output = t("The node has been updated.");
Dries's avatar
 
Dries committed
1015 1016
      }
      else {
Dries's avatar
 
Dries committed
1017 1018
        watchdog("warning", "$node->type: not authorized to update node");
        $output = t("You are not authorized to update this node.");
Dries's avatar
 
Dries committed
1019 1020
      }

Dries's avatar
 
Dries committed
1021 1022 1023
    }
    else {

Dries's avatar
 
Dries committed
1024 1025 1026 1027
      /*
      ** Check whether the current user has the proper access rights to
      ** perform this operation:
      */
Dries's avatar
 
Dries committed
1028

Dries's avatar
 
Dries committed
1029
      if (node_access("create", $node)) {
Dries's avatar
 
Dries committed
1030

Dries's avatar
 
Dries committed
1031 1032 1033 1034 1035 1036 1037
        /*
        ** Verify a user's submission rate and avoid duplicate nodes being
        ** inserted:
        */

        throttle("node", variable_get("max_node_rate", 900));

Dries's avatar
 
Dries committed
1038 1039 1040 1041
        /*
        ** 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
1042

Dries's avatar
 
Dries committed
1043 1044 1045 1046
        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
1047
          $fields = array("uid" => ($user->uid ? $user->uid : 0), "body", "comment" => 1, "teaser", "title", "type" => $node->type);
Dries's avatar
 
Dries committed
1048 1049
        }

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

Dries's avatar
 
Dries committed
1052 1053
        watchdog("special", "$node->type: added '$node->title'");
        $output = t("Thanks for your submission.");
Dries's avatar
 
Dries committed
1054 1055
      }
      else {
Dries's avatar
 
Dries committed
1056 1057
        watchdog("warning", "$node->type: not authorized to create node");
        $output = t("You are not authorized to create this node.");
Dries's avatar
 
Dries committed
1058 1059
      }
    }
Dries's avatar
 
Dries committed
1060

Dries's avatar
 
Dries committed
1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071
    /*
    ** Reload the node from the database:
    */

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

    /*
    ** For usability's sake, make sure to present the user with some</