HTNDir.c
上传用户:zlh9724
上传日期:2007-01-04
资源大小:1991k
文件大小:8k
源码类别:

浏览器

开发平台:

Unix_Linux

  1. /*              HTNDir.c
  2. ** GENERIC NEWS LISTINGS
  3. **
  4. ** (c) COPYRIGHT MIT 1995.
  5. ** Please first read the full copyright statement in the file COPYRIGH.
  6. **
  7. ** Creates listings for all kind of News output.
  8. **
  9. ** Authors:
  10. ** HF Henrik Frystyk, MIT, <frystyk@w3.org>
  11. ** History:
  12. **    Oct 95  HFN written
  13. */
  14. /* Library include files */
  15. #include "tcp.h"
  16. #include "HTUtils.h"
  17. #include "HTString.h"
  18. #include "HTMLPDTD.h"
  19. #include "HTMLGen.h"
  20. #include "HTEscape.h"
  21. #include "HTParse.h"
  22. #include "HTFormat.h"
  23. #include "HTReq.h"
  24. #include "HTStruct.h"
  25. #include "HTArray.h"
  26. #include "HTError.h"
  27. #include "HTNews.h"
  28. #include "HTNDir.h"  /* Implemented here */
  29. /* Macros and other defines */
  30. #define PUTC(c) (*target->isa->put_character)(target, c)
  31. #define PUTS(s) (*target->isa->put_string)(target, s)
  32. #define START(e) (*target->isa->start_element)(target, e, 0, 0)
  33. #define END(e) (*target->isa->end_element)(target, e)
  34. #define FREE_TARGET (*target->isa->_free)(target)
  35. #define DEFAULT_MAXW 80        /* Default line width */
  36. /* Type definitions and global variables etc. local to this module */
  37. struct _HTStructured {
  38.     CONST HTStructuredClass * isa;
  39.     /* ... */
  40. };
  41. struct _HTNewsDir {
  42.     HTStructured * target;
  43.     HTRequest * request;
  44.     HTArray * array; /* Array for sorted listings */
  45.     BOOL top; /* YES if first line */
  46.     HTNewsDirKey key;   /* Key for sorting */
  47. };
  48. typedef struct _HTNewsNode {
  49.     int index;
  50.     char * name;
  51.     char * subject;
  52.     char * from;
  53.     time_t date;
  54.     int refs;      /* Number of references */
  55.     BOOL filter;       /* Is this entry filtered out? */
  56. } HTNewsNode;
  57. PRIVATE int MaxLineW = DEFAULT_MAXW;
  58. /* ------------------------------------------------------------------------- */
  59. /* NODE  MANAGEMENT       */
  60. /* ------------------------------------------------------------------------- */
  61. /*
  62. ** Create a sort key node
  63. */
  64. PRIVATE HTNewsNode * HTNewsNode_new (void)
  65. {
  66.     HTNewsNode *node;
  67.     if ((node = (HTNewsNode *) HT_CALLOC(1, sizeof(HTNewsNode))) == NULL)
  68.         HT_OUTOFMEM("HTNewsNode_new");
  69.     return node;
  70. }
  71. /*
  72. ** Free a sort key node
  73. */
  74. PRIVATE BOOL HTNewsNode_free (HTNewsNode *node)
  75. {
  76.     if (node) {
  77. HT_FREE(node->name);
  78. HT_FREE(node->subject);
  79. HT_FREE(node->from);
  80. HT_FREE(node);
  81. return YES;
  82.     }
  83.     return NO;
  84. }
  85. /*
  86. ** Output an element in HTML
  87. ** Returns YES if OK, else NO
  88. */
  89. PRIVATE BOOL HTNewsNode_print (HTNewsDir *dir, HTNewsNode *node)
  90. {
  91.     HTStructured *target = dir->target;
  92.     START(HTML_LI);
  93.     /* Start the anchor and put the subject as anchor text */
  94.     if (node->name && node->subject) {
  95. char *escaped = HTEscape(node->name, URL_XPALPHAS);
  96. HTStartAnchor(target, NULL, escaped);
  97. PUTS(node->subject);
  98. END(HTML_A);
  99. HT_FREE(escaped);
  100.     }
  101.     /* From field */
  102.     if (node->from) {
  103. PUTC(' ');
  104. PUTS(node->from);
  105.     }
  106.     return YES;
  107. }
  108. /* ------------------------------------------------------------------------- */
  109. /* DIRECTORY MANAGEMENT       */
  110. /* ------------------------------------------------------------------------- */
  111. /* HTNewsDir_headLine
  112. ** ---------------
  113. **     Puts out the header line of the list itself
  114. ** Returns YES if OK, else NO
  115. */
  116. PRIVATE BOOL HTNewsDir_headLine (HTNewsDir *dir)
  117. {
  118.     if (dir) {
  119. HTStructured *target = dir->target;
  120. PUTS("HERE WE CAN PUT INFORMATION AND EXTRA LINKSn");
  121. return YES;
  122. START(HTML_UL);
  123.     }
  124.     return NO;
  125. }
  126. /* HTNewsDir_setWidth
  127. ** ------------------
  128. ** The module automatically ajusts the width of the directory listing as
  129. ** a function of the file name. The width can flows dynamically between
  130. ** an upper and a lower limit.
  131. */
  132. PUBLIC BOOL HTNewsDir_setWidth (int max_width)
  133. {
  134.     MaxLineW = (max_width > 0) ? max_width : DEFAULT_MAXW;
  135.     return YES;
  136. }
  137. /* HTNewsDir_new
  138. ** ----------
  139. **     Creates a structured stream object and sets up the initial HTML stuff
  140. ** Returns the newsdir object if OK, else NULL
  141. */
  142. PUBLIC HTNewsDir * HTNewsDir_new (HTRequest * request, CONST char * title,
  143.   HTNewsDirKey key)
  144. {
  145.     HTNewsDir *dir;
  146.     if (!request) return NULL;
  147.     /* Create object */
  148.     if ((dir = (HTNewsDir *) HT_CALLOC(1, sizeof (HTNewsDir))) == NULL)
  149.         HT_OUTOFMEM("HTNewsDir_new");
  150.     dir->target = HTMLGenerator(request, NULL, WWW_HTML,
  151. HTRequest_outputFormat(request),
  152. HTRequest_outputStream(request));
  153.     HTAnchor_setFormat(HTRequest_anchor(request), WWW_HTML);
  154.     dir->request = request;
  155.     dir->top = YES;
  156.     dir->key = key;
  157.     if (key != HT_NDK_NONE) {        /* Thread is unsorted */
  158. int total = HTNews_maxArticles();
  159. dir->array = HTArray_new(total > 0 ? total : 128);
  160.     }
  161.     /* Start the HTML stuff */
  162.     {
  163. HTStructured *target = dir->target;
  164. CONST char *msg = title ? title : "News Listing";
  165. START(HTML_HTML);
  166. START(HTML_HEAD);
  167. START(HTML_TITLE);
  168. PUTS(msg);
  169. END(HTML_TITLE);
  170. END(HTML_HEAD);
  171. START(HTML_BODY);
  172. START(HTML_H1);
  173. PUTS(msg);
  174. END(HTML_H1);
  175.     }
  176.     return dir;
  177. }
  178. /* HTNewsDir_addElement
  179. ** --------------------
  180. **     This function accepts a news line. Everything except dir and name can
  181. ** can be 0 or NULL.
  182. ** Returns YES if OK, else NO
  183. */
  184. PUBLIC BOOL HTNewsDir_addElement (HTNewsDir * dir, int index, char * subject,
  185.   char * from, time_t date, char * name,
  186.   int refs)
  187. {
  188.     HTNewsNode *node = HTNewsNode_new();
  189.     if (!dir || !name) return NO;
  190.     StrAllocCopy(node->name, name); /* Mandatory */
  191.     if (subject) StrAllocCopy(node->subject, subject);
  192.     if (from) StrAllocCopy(node->from, from);
  193.     node->date = date;
  194.     if (dir->key == HT_NDK_NONE) {
  195. if (dir->top) {
  196.     HTNewsDir_headLine(dir);
  197.     dir->top = NO;
  198. }
  199. HTNewsNode_print(dir, node);
  200. HTNewsNode_free(node);
  201.     } else
  202. HTArray_addObject(dir->array, (void *) node);
  203.     return YES;
  204. }
  205. PRIVATE int NDirIndexSort (CONST void *a, CONST void *b)
  206. {
  207.     int aa = (*((HTNewsNode **)a))->index;
  208.     int bb = (*((HTNewsNode **)b))->index;
  209.     return aa-bb;
  210. }
  211. PRIVATE int NDirSubjectSort (CONST void *a, CONST void *b)
  212. {
  213. #if 0
  214.     char *aa = (*((HTNewsNode **)a))->subject;
  215.     char *bb = (*((HTNewsNode **)b))->subject;
  216. #endif
  217.     return 0;
  218. }
  219. PRIVATE int NDirFromSort (CONST void *a, CONST void *b)
  220. {
  221. #if 0
  222.     HTNewsNode *aa = *(HTNewsNode **) a;
  223.     HTNewsNode *bb = *(HTNewsNode **) b;
  224.     return strcasecomp(aa->fname, bb->fname);
  225.     return strcasecomp((*(HTNewsNode**)a)->fname, (*(HTNewsNode**)a)->fname);
  226.     char *aa = (*((HTNewsNode **)a))->name;
  227.     char *bb = (*((HTNewsNode **)b))->name;
  228. #endif
  229.     return 1;
  230. }
  231. PRIVATE int NDirDateSort (CONST void *a, CONST void *b)
  232. {
  233.     time_t aa = (*((HTNewsNode **)a))->date;
  234.     time_t bb = (*((HTNewsNode **)b))->date;
  235.     return bb-aa;
  236. }
  237. PRIVATE int NDirGroupSort (CONST void *a, CONST void *b)
  238. {
  239.     char *aa = (*((HTNewsNode **)a))->name;
  240.     char *bb = (*((HTNewsNode **)b))->name;
  241.     while (*aa && *bb && TOLOWER(*aa)==TOLOWER(*bb)) aa++, bb++;
  242.     return (*aa=='.' && *bb) ? -1 : (*aa && *bb=='.') ?
  243. 1 : TOLOWER(*aa)-TOLOWER(*bb);
  244. }
  245. /* HTNewsDir_free
  246. ** --------------
  247. **     If we are sorting then do the sorting and put out the list,
  248. ** else just append the end of the list.
  249. */
  250. PUBLIC BOOL HTNewsDir_free (HTNewsDir * dir)
  251. {
  252.     if (!dir) return NO;
  253.     if (dir->key != HT_NDK_NONE) {
  254. HTArray *array = dir->array;
  255. HTComparer * comp = NULL;
  256. HTNewsDir_headLine(dir);
  257. if (dir->key == HT_NDK_INDEX)    /* Sort by Message Number */
  258.     comp = NDirIndexSort;
  259. if (dir->key == HT_NDK_DATE)              /* Sort by Date */
  260.     comp = NDirDateSort;
  261. if (dir->key == HT_NDK_SUBJECT)        /* Sort after Subject */
  262.     comp = NDirSubjectSort;
  263. else if (dir->key == HT_NDK_FROM)   /* Sort after From */
  264.     comp = NDirFromSort;
  265. else if (dir->key == HT_NDK_GROUP)   /* Sort as group hierarchi */
  266.     comp = NDirGroupSort;
  267. else {
  268.     if (STREAM_TRACE) TTYPrint(TDEST,"NewsListing. Invalid sortkeyn");
  269.     return NO;
  270. }
  271. HTArray_sort(array, comp);
  272. {
  273.     void **data;
  274.     HTNewsNode *node = (HTNewsNode *) HTArray_firstObject(array, data);
  275.     while (node) {
  276. HTNewsNode_print(dir, node);
  277. HTNewsNode_free(node);
  278. node = (HTNewsNode *) HTArray_nextObject(array, data);
  279.     }
  280. }
  281.      HTArray_delete(array);
  282.     }
  283.     /* Put out the end of the HTML stuff */
  284.     {
  285. HTStructured *target = dir->target;
  286. END(HTML_UL);
  287. START(HTML_HR);
  288. END(HTML_BODY);
  289. END(HTML_HTML);
  290. FREE_TARGET;
  291.     }
  292.     HT_FREE(dir);
  293.     return YES;
  294. }