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

浏览器

开发平台:

Unix_Linux

  1. /*       HTStyle.c
  2. ** STYLE IMPLEMENTATION FOR HYPERTEXT
  3. **
  4. ** (c) COPYRIGHT MIT 1995.
  5. ** Please first read the full copyright statement in the file COPYRIGH.
  6. **
  7. ** Styles allow the translation between a logical property
  8. ** of a piece of text and its physical representation.
  9. **
  10. ** A StyleSheet is a collection of styles, defining the
  11. ** translation necessary to
  12. ** represent a document. It is a linked list of styles.
  13. */
  14. /* Library include files */
  15. #include "tcp.h"
  16. #include "HTUtils.h"
  17. #include "HTString.h"
  18. #include "HTStyle.h"
  19. /* Local definition of style
  20. ** -------------------------
  21. */
  22. /*      The Style Structure
  23. **      -------------------
  24. */
  25. typedef double HTCoord;
  26. typedef int HTColor;
  27. typedef struct {
  28.     short               kind;           /* only NX_LEFTTAB implemented*/
  29.     HTCoord             position;       /* x coordinate for stop */
  30. } HTTabStop;
  31. struct _HTStyle {
  32. /*      Style management information
  33. */
  34.     struct _HTStyle     *next;          /* Link for putting into stylesheet */
  35.     char *              name;           /* Style name */
  36.     char *              SGMLTag;        /* Tag name to start */
  37. /*      Character attributes    (a la NXRun)
  38. */
  39.     HTFont              font;           /* Font id */
  40.     HTCoord             fontSize;       /* The size of font, not independent */
  41.     HTColor             color;  /* text gray of current run */
  42.     int                 superscript;    /* superscript (-sub) in points */
  43.     HTAnchor            *anchor;        /* Anchor id if any, else zero */
  44. /*      Paragraph Attribtes     (a la NXTextStyle)
  45. */
  46.     HTCoord             indent1st;      /* how far first line in paragraph is
  47.                                  * indented */
  48.     HTCoord             leftIndent;     /* how far second line is indented */
  49.     HTCoord             rightIndent;    /* (Missing from NeXT version */
  50.     short               alignment;      /* quad justification */
  51.     HTCoord             lineHt;         /* line height */
  52.     HTCoord             descentLine;    /* descender bottom from baseline */
  53.     HTTabStop           *tabs;          /* array of tab stops, 0 terminated */
  54.     BOOL                wordWrap;       /* Yes means wrap at space not char */
  55.     BOOL                freeFormat;     /* Yes means n is just white space */
  56.     HTCoord             spaceBefore;    /* Omissions from NXTextStyle */
  57.     HTCoord             spaceAfter;
  58.     int                 paraFlags;      /* Paragraph flags, bits as follows: */
  59. #define PARA_KEEP       1       /* Do not break page within this paragraph */
  60. #define PARA_WITH_NEXT  2       /* Do not break page after this paragraph */
  61. #define HT_JUSTIFY 0            /* For alignment */
  62. #define HT_LEFT 1
  63. #define HT_RIGHT 2
  64. #define HT_CENTER 3
  65. };
  66. /* Create a new style
  67. */
  68. PUBLIC HTStyle* HTStyleNew (void)
  69. {
  70.     HTStyle *style;
  71.     if ((style = (HTStyle  *) HT_CALLOC(1, sizeof(HTStyle))) == NULL)
  72.         HT_OUTOFMEM("HTStyleNew");
  73.     return style;
  74. }
  75. /* Create a new style with a name
  76. */
  77. PUBLIC HTStyle* HTStyleNewNamed  (CONST char *name)
  78. {
  79.     HTStyle * self = HTStyleNew();
  80.     StrAllocCopy(self->name, name);
  81.     return self;
  82. }
  83. /* Free a style
  84. */
  85. PUBLIC HTStyle * HTStyleFree  (HTStyle *self)
  86. {
  87.     if (self->name) HT_FREE(self->name);
  88.     if (self->SGMLTag) HT_FREE(self->SGMLTag);
  89.     HT_FREE(self);
  90.     return 0;
  91. }
  92. #ifdef SUPPRESS  /* Only on the NeXT */
  93. /* Read a style from a stream (without its name)
  94. ** --------------------------
  95. **
  96. ** Reads a style with paragraph information from a stream.
  97. ** The style name is not read or written by these routines.
  98. */
  99. #define NONE_STRING "(None)"
  100. #define HTStream NXStream
  101. HTStyle * HTStyleRead (HTStyle * style, HTStream * stream)
  102. {
  103.     char myTag[STYLE_NAME_LENGTH];
  104.     char fontName[STYLE_NAME_LENGTH];
  105.     NXTextStyle *p;
  106.     int tab;
  107.     int gotpara; /* flag: have we got a paragraph definition? */
  108.     NXScanf(stream, "%s%s%f%d",
  109. myTag,
  110. fontName,
  111. &style->fontSize,
  112. &gotpara);
  113.     if (gotpara) {
  114. if (!style->paragraph) {
  115.     if ((style->paragraph = HT_MALLOC(sizeof(*(style->paragraph)))) == NULL)
  116. HT_OUTOFMEM("paragraph");
  117.     style->paragraph->tabs = 0;
  118. }
  119. p = style->paragraph;
  120. NXScanf(stream, "%f%f%f%f%hd%f%f%hd",
  121.     &p->indent1st,
  122.     &p->indent2nd,
  123.     &p->lineHt,
  124.     &p->descentLine,
  125.     &p->alignment,
  126.     &style->spaceBefore,
  127.     &style->spaceAfter,
  128.     &p->numTabs);
  129. if (p->tabs) HT_FREE(p->tabs);
  130. if ((p->tabs = HT_MALLOC(p->numTabs * sizeof(p->tabs[0]))) == NULL)
  131.     HT_OUTOFMEM("tabs");
  132. for (tab=0; tab < p->numTabs; tab++) {
  133.     NXScanf(stream, "%hd%f",
  134.     &p->tabs[tab].kind,
  135.     &p->tabs[tab].x);
  136. }
  137.     } else { /* No paragraph */
  138.         if (style->paragraph) {
  139.          HT_FREE(style->paragraph);
  140.          style->paragraph = 0;
  141. }
  142.     } /* if no paragraph */
  143.     StrAllocCopy(style->SGMLTag, myTag);
  144.     if (strcmp(fontName, NONE_STRING)==0)
  145.         style->font = 0;
  146.     else
  147.         style->font = [Font newFont:fontName size:style->fontSize];
  148.     return 0;
  149. }
  150. /* Write a style to a stream in a compatible way
  151. */
  152. HTStyle * HTStyleWrite (HTStyle * style, NXStream * stream)
  153. {
  154.     int tab;
  155.     NXTextStyle *p = style->paragraph;
  156.     NXPrintf(stream, "%s %s %f %dn",
  157. style->SGMLTag,
  158. style->font ? [style->font name] : NONE_STRING,
  159. style->fontSize,
  160. p!=0);
  161.     if (p) {
  162. NXPrintf(stream, "t%f %f %f %f %d %f %ft%dn",
  163.     p->indent1st,
  164.     p->indent2nd,
  165.     p->lineHt,
  166.     p->descentLine,
  167.     p->alignment,
  168.     style->spaceBefore,
  169.     style->spaceAfter,
  170.     p->numTabs);
  171.     
  172. for (tab=0; tab < p->numTabs; tab++)
  173.     NXPrintf(stream, "t%d %fn",
  174.     p->tabs[tab].kind,
  175.     p->tabs[tab].x);
  176. }
  177.     return style;
  178. }
  179. /* Write a style to stdout for diagnostics
  180. */
  181. HTStyle * HTStyleDump (HTStyle * style)
  182. {
  183.     int tab;
  184.     NXTextStyle *p = style->paragraph;
  185.     TTYPrint(TDEST, "Style %d `%s' SGML:%s. Font %s %.1f point.n",
  186.      style,
  187. style->name,
  188. style->SGMLTag,
  189. [style->font name],
  190. style->fontSize);
  191.     if (p) {
  192.         TTYPrint(TDEST, 
  193.      "tIndents: first=%.0f others=%.0f, Height=%.1f Desc=%.1fn"
  194. "tAlign=%d, %d tabs. (%.0f before, %.0f after)n",
  195.     p->indent1st,
  196.     p->indent2nd,
  197.     p->lineHt,
  198.     p->descentLine,
  199.     p->alignment,
  200.     p->numTabs,
  201.     style->spaceBefore,
  202.     style->spaceAfter);
  203.     
  204. for (tab=0; tab < p->numTabs; tab++) {
  205.     TTYPrint(TDEST, "ttTab kind=%d at %.0fn",
  206.     p->tabs[tab].kind,
  207.     p->tabs[tab].x);
  208.      }
  209. TTYPrint(TDEST, "n");
  210.     } /* if paragraph */
  211.     return style;
  212. }
  213. #endif
  214. /* StyleSheet Functions
  215. ** ====================
  216. */
  217. /* Searching for styles:
  218. */
  219. HTStyle * HTStyleNamed  (HTStyleSheet *self, CONST char *name)
  220. {
  221.     if (self && name) {      /* added by HWL 11/8/94 */
  222. HTStyle * scan;
  223. for (scan=self->styles; scan; scan=scan->next)
  224.     if (!strcmp(scan->name, name)) return scan;
  225. if (SGML_TRACE)
  226.     TTYPrint(TDEST, "StyleSheet.. No style named `%s'n", name);
  227.     }
  228.     return NULL;
  229. }
  230. #ifdef NEXT_SUPRESS /* Not in general common code */
  231. HTStyle * HTStyleMatching (HTStyleSheet * self, HTStyle *style)
  232. {
  233.     HTStyle * scan;
  234.     for (scan=self->styles; scan; scan=scan->next)
  235.         if (scan->paragraph == para) return scan;
  236.     return 0;
  237. }
  238. /* Find the style which best fits a given run
  239. ** ------------------------------------------
  240. **
  241. ** This heuristic is used for guessing the style for a run of
  242. ** text which has been pasted in. In order, we try:
  243. **
  244. ** A style whose paragraph structure is actually used by the run.
  245. ** A style matching in font
  246. ** A style matching in paragraph style exactly
  247. ** A style matching in paragraph to a degree
  248. */
  249. HTStyle * HTStyleForRun (HTStyleSheet *self, NXRun *run)
  250. {
  251.     HTStyle * scan;
  252.     HTStyle * best = 0;
  253.     int bestMatch = 0;
  254.     NXTextStyle * rp = run->paraStyle;
  255.     for (scan=self->styles; scan; scan=scan->next)
  256.         if (scan->paragraph == run->paraStyle) return scan; /* Exact */
  257.     for (scan=self->styles; scan; scan=scan->next){
  258.      NXTextStyle * sp = scan->paragraph;
  259.      if (sp) {
  260.     int match = 0;
  261.     if (sp->indent1st == rp->indent1st) match = match+1;
  262.     if (sp->indent2nd == rp->indent2nd) match = match+2;
  263.     if (sp->lineHt == rp->lineHt) match = match+1;
  264.     if (sp->numTabs == rp->numTabs) match = match+1;
  265.     if (sp->alignment == rp->alignment) match = match+3;
  266.     if (scan->font == run->font) match = match+10;
  267.     if (match>bestMatch) {
  268.     best=scan;
  269.     bestMatch=match;
  270.     }
  271. }
  272.     }
  273.     if (WWWTRACE) TTYPrint(TDEST, "HTStyleForRun: Best match for style is %d out of 18n",
  274.       bestMatch);
  275.     return best;
  276. }
  277. #endif
  278. /* Add a style to a sheet
  279. ** ----------------------
  280. */
  281. HTStyleSheet * HTStyleSheetAddStyle 
  282.   (HTStyleSheet *self, HTStyle *style)
  283. {
  284.     style->next = 0; /* The style will go on the end */
  285.     if (!self->styles) {
  286.      self->styles = style;
  287.     } else {
  288.      HTStyle * scan;
  289.         for(scan=self->styles; scan->next; scan=scan->next); /* Find end */
  290. scan->next=style;
  291.     }
  292.     return self;
  293. }
  294. /* Remove the given object from a style sheet if it exists
  295. */
  296. HTStyleSheet * HTStyleSheetRemoveStyle 
  297.   (HTStyleSheet *self, HTStyle *style)
  298. {
  299.     if (self->styles == style) {
  300.      self->styles = style->next;
  301. return self;
  302.     } else {
  303.      HTStyle * scan;
  304. for(scan = self->styles; scan; scan = scan->next) {
  305.     if (scan->next == style) {
  306.         scan->next = style->next;
  307. return self;
  308.     }
  309. }
  310.     }
  311.     return 0;
  312. }
  313. /* Create new style sheet
  314. */
  315. HTStyleSheet * HTStyleSheetNew (void)
  316. {
  317.     HTStyleSheet * style;
  318.     if ((style = (HTStyleSheet  *) HT_CALLOC(1, sizeof(HTStyleSheet))) == NULL)
  319.         HT_OUTOFMEM("HTStyleSheetNew");
  320.     return style;
  321. }
  322. /* Free off a style sheet pointer
  323. */
  324. HTStyleSheet * HTStyleSheetFree  (HTStyleSheet *self)
  325. {
  326.     HTStyle * style;
  327.     while((style=self->styles)!=0) {
  328.         self->styles = style->next;
  329. HTStyleFree(style);
  330.     }
  331.     HT_FREE(self);
  332.     return 0;
  333. }
  334. /* Read a stylesheet from a typed stream
  335. ** -------------------------------------
  336. **
  337. ** Reads a style sheet from a stream.  If new styles have the same names
  338. ** as existing styles, they replace the old ones without changing the ids.
  339. */
  340. #ifdef NEXT_SUPRESS  /* Only on the NeXT */
  341. HTStyleSheet * HTStyleSheetRead(HTStyleSheet * self, NXStream * stream)
  342. {
  343.     int numStyles;
  344.     int i;
  345.     HTStyle * style;
  346.     char styleName[80];
  347.     NXScanf(stream, " %d ", &numStyles);
  348.     if (WWWTRACE) TTYPrint(TDEST, "Stylesheet: Reading %d stylesn", numStyles);
  349.     for (i=0; i<numStyles; i++) {
  350.         NXScanf(stream, "%s", styleName);
  351.         style = HTStyleNamed(self, styleName);
  352. if (!style) {
  353.     style = HTStyleNewNamed(styleName);
  354.     (void) HTStyleSheetAddStyle(self, style);
  355. }
  356. (void) HTStyleRead(style, stream);
  357. if (WWWTRACE) HTStyleDump(style);
  358.     }
  359.     return self;
  360. }
  361. /* Write a stylesheet to a typed stream
  362. ** ------------------------------------
  363. **
  364. ** Writes a style sheet to a stream.
  365. */
  366. HTStyleSheet * HTStyleSheetWrite(HTStyleSheet * self, NXStream * stream)
  367. {
  368.     int numStyles = 0;
  369.     HTStyle * style;
  370.     
  371.     for(style=self->styles; style; style=style->next) numStyles++;
  372.     NXPrintf(stream, "%dn", numStyles);
  373.     
  374.     if (WWWTRACE) TTYPrint(TDEST, "StyleSheet: Writing %d stylesn", numStyles);
  375.     for (style=self->styles; style; style=style->next) {
  376.         NXPrintf(stream, "%s ", style->name);
  377. (void) HTStyleWrite(style, stream);
  378.     }
  379.     return self;
  380. }
  381. #endif