X-Git-Url: https://code.citadel.org/?a=blobdiff_plain;f=webcit%2Fmsg_renderers.c;h=eeccd46ee121460e56d9f5d48bdcb65bfe241736;hb=c3cd364e864859835ebfe6e0aeda2bb07160063f;hp=e7d13ff10a0d2d7760595941a903bb8718ae95a5;hpb=baafaeb404f5ca8ec6c9d5d9cc47eb92560d0fbe;p=citadel.git diff --git a/webcit/msg_renderers.c b/webcit/msg_renderers.c index e7d13ff10..eeccd46ee 100644 --- a/webcit/msg_renderers.c +++ b/webcit/msg_renderers.c @@ -1,8 +1,7 @@ -/*----------------------------------------------------------------------------*/ #include "webcit.h" #include "webserver.h" -/** +/* * message index functions */ @@ -462,8 +461,10 @@ void tmplput_MAIL_SUMM_DATE_NO(StrBuf *Target, WCTemplputParams *TP) void render_MAIL(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset) { + const StrBuf *TemplateMime; + Mime->Data = NewStrBufPlain(NULL, Mime->length); - read_message(Mime->Data, HKEY("view_submessage"), Mime->msgnum, 0, Mime->PartNum); + read_message(Mime->Data, HKEY("view_submessage"), Mime->msgnum, Mime->PartNum, &TemplateMime); /* if ( (!IsEmptyStr(mime_submessages)) && (!section[0]) ) { for (i=0; iData), 0, 1, NULL, Mime->msgnum); + display_vcard(Buf, Mime->Data, 0, 1, NULL, Mime->msgnum); FreeStrBuf(&Mime->Data); Mime->Data = Buf; } } +void render_MIME_VNote(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset) +{ + MimeLoadData(Mime); + if (StrLength(Mime->Data) > 0) { + struct vnote *v; + StrBuf *Buf; + char *vcard; + + Buf = NewStrBuf(); + vcard = SmashStrBuf(&Mime->Data); + v = vnote_new_from_str(vcard); + free (vcard); + if (v) { + WCTemplputParams TP; + + memset(&TP, 0, sizeof(WCTemplputParams)); + TP.Filter.ContextType = CTX_VNOTE; + TP.Context = v; + DoTemplate(HKEY("mail_vnoteitem"), + Buf, &TP); + + vnote_free(v); + Mime->Data = Buf; + } + else + Mime->Data = NewStrBuf(); + } + +} + void render_MIME_ICS(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset) { if (StrLength(Mime->Data) == 0) { @@ -643,7 +674,7 @@ int Conditional_MAIL_SUMM_H_NODE(StrBuf *Target, WCTemplputParams *TP) void examine_text(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset) { - Msg->MsgBody->Data = NewStrBuf(); + Msg->MsgBody->Data = NewStrBufPlain(NULL, SIZ); } void examine_msg4_partnum(message_summary *Msg, StrBuf *HdrLine, StrBuf *FoundCharset) @@ -741,17 +772,28 @@ int Conditional_MAIL_MIME_ATTACH(StrBuf *Target, WCTemplputParams *TP) return GetCount(Msg->AllAttach) > 0; } +void tmplput_QUOTED_MAIL_BODY(StrBuf *Target, WCTemplputParams *TP) +{ + const StrBuf *Mime; + long MsgNum; + StrBuf *Buf; + MsgNum = LBstr(TKEY(0)); + Buf = NewStrBuf(); + read_message(Buf, HKEY("view_message_replyquote"), MsgNum, NULL, &Mime); + StrBufAppendTemplate(Target, TP, Buf, 1); + FreeStrBuf(&Buf); +} -/*----------------------------------------------------------------------------*/ -void tmplput_QUOTED_MAIL_BODY(StrBuf *Target, WCTemplputParams *TP) +void tmplput_EDIT_MAIL_BODY(StrBuf *Target, WCTemplputParams *TP) { + const StrBuf *Mime; long MsgNum; StrBuf *Buf; MsgNum = LBstr(TKEY(0)); Buf = NewStrBuf(); - read_message(Buf, HKEY("view_message_replyquote"), MsgNum, 0, NULL); + read_message(Buf, HKEY("view_message_edit"), MsgNum, NULL, &Mime); StrBufAppendTemplate(Target, TP, Buf, 1); FreeStrBuf(&Buf); } @@ -774,7 +816,6 @@ void render_MAIL_variformat(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *F void render_MAIL_text_plain(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *FoundCharset) { - StrBuf *cs = NULL; const char *ptr, *pte; const char *BufPtr = NULL; StrBuf *Line; @@ -782,12 +823,13 @@ void render_MAIL_text_plain(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *F StrBuf *Line2; StrBuf *Target; - int ConvertIt = 1; int bn = 0; int bq = 0; - int i, n, done = 0; + int i; long len; #ifdef HAVE_ICONV + StrBuf *cs = NULL; + int ConvertIt = 1; iconv_t ic = (iconv_t)(-1) ; #endif @@ -796,11 +838,6 @@ void render_MAIL_text_plain(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *F MimeLoadData(Mime); } - /* Boring old 80-column fixed format text gets handled this way... */ - if ((strcasecmp(ChrPtr(Mime->Charset), "us-ascii") == 0) && - (strcasecmp(ChrPtr(Mime->Charset), "UTF-8") == 0)) - ConvertIt = 0; - #ifdef HAVE_ICONV if (ConvertIt) { if (StrLength(Mime->Charset) != 0) @@ -812,6 +849,12 @@ void render_MAIL_text_plain(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *F if (cs == NULL) { ConvertIt = 0; } + else if (!strcasecmp(ChrPtr(cs), "utf-8")) { + ConvertIt = 0; + } + else if (!strcasecmp(ChrPtr(cs), "us-ascii")) { + ConvertIt = 0; + } else { ctdl_iconv_open("UTF-8", ChrPtr(cs), &ic); if (ic == (iconv_t)(-1) ) { @@ -821,52 +864,55 @@ void render_MAIL_text_plain(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *F } } #endif - Line = NewStrBuf(); - Line1 = NewStrBuf(); - Line2 = NewStrBuf(); + Line = NewStrBufPlain(NULL, SIZ); + Line1 = NewStrBufPlain(NULL, SIZ); + Line2 = NewStrBufPlain(NULL, SIZ); Target = NewStrBufPlain(NULL, StrLength(Mime->Data)); - while ((n = StrBufSipLine(Line, Mime->Data, &BufPtr), n >= 0) && !done) - { - done = n == 0; - bq = 0; - i = 0; - ptr = ChrPtr(Line); - len = StrLength(Line); - pte = ptr + len; - - while ((ptr < pte) && - ((*ptr == '>') || - isspace(*ptr))) + if (StrLength(Mime->Data) > 0) + do { - if (*ptr == '>') - bq++; - ptr ++; - i++; - } - if (i > 0) StrBufCutLeft(Line, i); + StrBufSipLine(Line, Mime->Data, &BufPtr); + bq = 0; + i = 0; + ptr = ChrPtr(Line); + len = StrLength(Line); + pte = ptr + len; - if (StrLength(Line) == 0) { - StrBufAppendBufPlain(Target, HKEY("
\n"), 0); - continue; - } + while ((ptr < pte) && + ((*ptr == '>') || + isspace(*ptr))) + { + if (*ptr == '>') + bq++; + ptr ++; + i++; + } + if (i > 0) StrBufCutLeft(Line, i); + + if (StrLength(Line) == 0) { + StrBufAppendBufPlain(Target, HKEY("
\n"), 0); + continue; + } - for (i = bn; i < bq; i++) - StrBufAppendBufPlain(Target, HKEY("
"), 0); - for (i = bq; i < bn; i++) - StrBufAppendBufPlain(Target, HKEY("
"), 0); + for (i = bn; i < bq; i++) + StrBufAppendBufPlain(Target, HKEY("
"), 0); + for (i = bq; i < bn; i++) + StrBufAppendBufPlain(Target, HKEY("
"), 0); +#ifdef HAVE_ICONV + if (ConvertIt) { + StrBufConvert(Line, Line1, &ic); + } +#endif + StrBufAppendBufPlain(Target, HKEY(""), 0); + UrlizeText(Line1, Line, Line2); - if (ConvertIt == 1) { - StrBufConvert(Line, Line1, &ic); + StrEscAppend(Target, Line1, NULL, 0, 0); + StrBufAppendBufPlain(Target, HKEY("
\n"), 0); + bn = bq; } - - StrBufAppendBufPlain(Target, HKEY(""), 0); - UrlizeText(Line1, Line, Line2); - - StrEscAppend(Target, Line1, NULL, 0, 0); - StrBufAppendBufPlain(Target, HKEY("
\n"), 0); - bn = bq; - } + while ((BufPtr != StrBufNOTNULL) && + (BufPtr != NULL)); for (i = 0; i < bn; i++) StrBufAppendBufPlain(Target, HKEY(""), 0); @@ -917,10 +963,6 @@ void render_MAIL_UNKNOWN(wc_mime_attachment *Mime, StrBuf *RawData, StrBuf *Foun } - - - - HashList *iterate_get_mime_All(StrBuf *Target, WCTemplputParams *TP) { message_summary *Msg = (message_summary*) CTX; @@ -1032,6 +1074,98 @@ void tmplput_MIME_Length(StrBuf *Target, WCTemplputParams *TP) StrBufAppendPrintf(Target, "%ld", mime->length); } +/* startmsg is an index within the message list. + * starting_from is the Citadel message number to be supplied to a "MSGS GT" operation + */ +long DrawMessageDropdown(StrBuf *Selector, long maxmsgs, long startmsg, int nMessages, long starting_from) +{ + StrBuf *TmpBuf; + wcsession *WCC = WC; + void *vMsg; + int lo, hi; + long ret; + long hklen; + const char *key; + int done = 0; + int nItems; + HashPos *At; + long vector[16]; + WCTemplputParams SubTP; + + memset(&SubTP, 0, sizeof(WCTemplputParams)); + SubTP.Filter.ContextType = CTX_LONGVECTOR; + SubTP.Context = &vector; + TmpBuf = NewStrBufPlain(NULL, SIZ); + At = GetNewHashPos(WCC->summ, nMessages); + nItems = GetCount(WCC->summ); + ret = nMessages; + vector[0] = 7; + vector[2] = 1; + vector[1] = startmsg; + vector[3] = 0; + vector[7] = starting_from; + + while (!done) { + vector[3] = abs(nMessages); + lo = GetHashPosCounter(At); + if (nMessages > 0) { + if (lo + nMessages >= nItems) { + hi = nItems - 1; + vector[3] = nItems - lo; + if (startmsg == lo) + ret = vector[3]; + } + else { + hi = lo + nMessages - 1; + } + } else { + if (lo + nMessages < -1) { + hi = 0; + } + else { + if ((lo % abs(nMessages)) != 0) { + int offset = (lo % abs(nMessages) * + (nMessages / abs(nMessages))); + hi = lo + offset; + vector[3] = abs(offset); + if (startmsg == lo) + ret = offset; + } + else + hi = lo + nMessages; + } + } + done = !GetNextHashPos(WCC->summ, At, &hklen, &key, &vMsg); + + /* + * Bump these because although we're thinking in zero base, the user + * is a drooling idiot and is thinking in one base. + */ + vector[4] = lo + 1; + vector[5] = hi + 1; + vector[6] = lo; + FlushStrBuf(TmpBuf); + dbg_print_longvector(vector); + DoTemplate(HKEY("select_messageindex"), TmpBuf, &SubTP); + StrBufAppendBuf(Selector, TmpBuf, 0); + } + vector[6] = 0; + FlushStrBuf(TmpBuf); + if (maxmsgs == 9999999) { + vector[1] = 1; + ret = maxmsgs; + } + else + vector[1] = 0; + vector[2] = 0; + dbg_print_longvector(vector); + DoTemplate(HKEY("select_messageindex_all"), TmpBuf, &SubTP); + StrBufAppendBuf(Selector, TmpBuf, 0); + FreeStrBuf(&TmpBuf); + DeleteHashPos(&At); + return ret; +} + HashList *iterate_get_registered_Attachments(StrBuf *Target, WCTemplputParams *TP) { return WC->attachments; @@ -1052,6 +1186,11 @@ void servcmd_readfwd(char *buf, long bufsize) snprintf(buf, bufsize, "MSGS ALL"); } +void servcmd_readgt(char *buf, long bufsize) +{ + snprintf(buf, bufsize, "MSGS GT|%s", bstr("gt")); +} + void servcmd_readnew(char *buf, long bufsize) { snprintf(buf, bufsize, "MSGS NEW"); @@ -1068,32 +1207,248 @@ readloop_struct rlid[] = { { {HKEY("headers")}, servcmd_headers}, { {HKEY("readfwd")}, servcmd_readfwd}, { {HKEY("readnew")}, servcmd_readnew}, - { {HKEY("readold")}, servcmd_readold} + { {HKEY("readold")}, servcmd_readold}, + { {HKEY("readgt")}, servcmd_readgt} }; +/* Spit out the new summary view. This is basically a static page, so clients can cache the layout, all the dirty work is javascript :) */ +void new_summary_view(void) { + DoTemplate(HKEY("msg_listview"),NULL,&NoCtx); + DoTemplate(HKEY("trailing"),NULL,&NoCtx); +} + + +int mailview_GetParamsGetServerCall(SharedMessageStatus *Stat, + void **ViewSpecific, + long oper, + char *cmd, + long len) +{ + if (!WC->is_ajax) { + new_summary_view(); + return 200; + } else { + Stat->defaultsortorder = 2; + Stat->sortit = 1; + Stat->load_seen = 1; + /* Generally using maxmsgs|startmsg is not required + in mailbox view, but we have a 'safemode' for clients + (*cough* Exploder) that simply can't handle too many */ + if (havebstr("maxmsgs")) Stat->maxmsgs = ibstr("maxmsgs"); + else Stat->maxmsgs = 9999999; + if (havebstr("startmsg")) Stat->startmsg = lbstr("startmsg"); + snprintf(cmd, len, "MSGS %s|%s||1", + (oper == do_search) ? "SEARCH" : "ALL", + (oper == do_search) ? bstr("query") : "" + ); + } + return 200; +} +int mailview_RenderView_or_Tail(SharedMessageStatus *Stat, + void **ViewSpecific, + long oper) +{ + WCTemplputParams SubTP; -void SetAccessCommand(long Oper) + DoTemplate(HKEY("mailsummary_json"),NULL, &SubTP); + return 0; +} + +int mailview_Cleanup(void **ViewSpecific) { - wcsession *WCC = WC; + /* Note: wDumpContent() will output one additional tag. */ + /* We ought to move this out into template */ + if (WC->is_ajax) + end_burst(); + else + wDumpContent(1); + return 0; +} - if (WCC->UrlFragment1 != NULL ) { - FlushStrBuf(WCC->UrlFragment1); - StrBufAppendBufPlain(WCC->UrlFragment1, - rlid[Oper].name.Key, rlid[Oper].name.len, 0); + + +typedef struct _bbsview_stuct { + StrBuf *BBViewToolBar; + StrBuf *MessageDropdown; + long *displayed_msgs; + int a; +}bbsview_struct; + +int bbsview_GetParamsGetServerCall(SharedMessageStatus *Stat, + void **ViewSpecific, + long oper, + char *cmd, + long len) +{ + bbsview_struct *VS; + + VS = (bbsview_struct*) malloc(sizeof(bbsview_struct)); + memset(VS, 0, sizeof(bbsview_struct)); + *ViewSpecific = (void*)VS; + Stat->defaultsortorder = 1; + Stat->startmsg = -1; + Stat->sortit = 1; + + rlid[oper].cmd(cmd, len); + + if (havebstr("maxmsgs")) + Stat->maxmsgs = ibstr("maxmsgs"); + if (Stat->maxmsgs == 0) Stat->maxmsgs = DEFAULT_MAXMSGS; + + if (havebstr("startmsg")) { + Stat->startmsg = lbstr("startmsg"); + } + if (lbstr("SortOrder") == 2) { + Stat->reverse = 1; + Stat->num_displayed = -DEFAULT_MAXMSGS; + } + else { + Stat->reverse = 0; + Stat->num_displayed = DEFAULT_MAXMSGS; + } + + return 200; +} + +int bbsview_PrintViewHeader(SharedMessageStatus *Stat, void **ViewSpecific) +{ + bbsview_struct *VS; + WCTemplputParams SubTP; + + VS = (bbsview_struct*)*ViewSpecific; + + VS->BBViewToolBar = NewStrBufPlain(NULL, SIZ); + VS->MessageDropdown = NewStrBufPlain(NULL, SIZ); + + /*** startmsg->maxmsgs = **/DrawMessageDropdown(VS->MessageDropdown, + Stat->maxmsgs, + Stat->startmsg, + Stat->num_displayed, + Stat->lowest_found-1); + if (Stat->num_displayed < 0) { + Stat->startmsg += Stat->maxmsgs; + if (Stat->num_displayed != Stat->maxmsgs) + Stat->maxmsgs = abs(Stat->maxmsgs) + 1; + else + Stat->maxmsgs = abs(Stat->maxmsgs); + + } + if (Stat->nummsgs > 0) { + memset(&SubTP, 0, sizeof(WCTemplputParams)); + SubTP.Filter.ContextType = CTX_STRBUF; + SubTP.Context = VS->MessageDropdown; + DoTemplate(HKEY("msg_listselector_top"), VS->BBViewToolBar, &SubTP); + StrBufAppendBuf(WC->WBuf, VS->BBViewToolBar, 0); + FlushStrBuf(VS->BBViewToolBar); + } + return 200; +} + +int bbsview_LoadMsgFromServer(SharedMessageStatus *Stat, + void **ViewSpecific, + message_summary* Msg, + int is_new, + int i) +{ + bbsview_struct *VS; + + VS = (bbsview_struct*)*ViewSpecific; + if (VS->displayed_msgs == NULL) { + VS->displayed_msgs = malloc(sizeof(long) * + ((Stat->maxmsgs < Stat->nummsgs) ? + Stat->maxmsgs + 1 : + Stat->nummsgs + 1)); + } + if ((i >= Stat->startmsg) && (i < Stat->startmsg + Stat->maxmsgs)) { + VS->displayed_msgs[Stat->num_displayed] = Msg->msgnum; + Stat->num_displayed++; + } + return 200; +} + + +int bbsview_RenderView_or_Tail(SharedMessageStatus *Stat, + void **ViewSpecific, + long oper) +{ + wcsession *WCC = WC; + bbsview_struct *VS; + WCTemplputParams SubTP; + const StrBuf *Mime; + + VS = (bbsview_struct*)*ViewSpecific; + if (Stat->nummsgs == 0) { + wprintf("

"); + switch (oper) { + case readnew: + wprintf(_("No new messages.")); + break; + case readold: + wprintf(_("No old messages.")); + break; + default: + wprintf(_("No messages here.")); + } + wprintf("
\n"); } else - WCC->UrlFragment1 = NewStrBufPlain(rlid[Oper].name.Key, rlid[Oper].name.len); + { + if (VS->displayed_msgs != NULL) { + /* if we do a split bbview in the future, begin messages div here */ + int a;/// todo + for (a=0; a < Stat->num_displayed; ++a) { + read_message(WCC->WBuf, HKEY("view_message"), VS->displayed_msgs[a], NULL, &Mime); + } + + /* if we do a split bbview in the future, end messages div here */ + + free(VS->displayed_msgs); + VS->displayed_msgs = NULL; + } + memset(&SubTP, 0, sizeof(WCTemplputParams)); + SubTP.Filter.ContextType = CTX_STRBUF; + SubTP.Context = VS->MessageDropdown; + DoTemplate(HKEY("msg_listselector_bottom"), VS->BBViewToolBar, &SubTP); + StrBufAppendBuf(WCC->WBuf, VS->BBViewToolBar, 0); + } + return 0; + } - +int bbsview_Cleanup(void **ViewSpecific) +{ + bbsview_struct *VS; + VS = (bbsview_struct*)*ViewSpecific; + end_burst(); + FreeStrBuf(&VS->BBViewToolBar); + FreeStrBuf(&VS->MessageDropdown); + free(VS); + return 0; +} void InitModule_MSGRENDERERS (void) { + RegisterReadLoopHandlerset( + VIEW_MAILBOX, + mailview_GetParamsGetServerCall, + NULL, /// TODO: is this right? + NULL, //// "" + mailview_RenderView_or_Tail, + mailview_Cleanup); + + RegisterReadLoopHandlerset( + VIEW_BBS, + bbsview_GetParamsGetServerCall, + bbsview_PrintViewHeader, + bbsview_LoadMsgFromServer, + bbsview_RenderView_or_Tail, + bbsview_Cleanup); + RegisterSortFunc(HKEY("date"), NULL, 0, summcmp_date, @@ -1136,6 +1491,7 @@ InitModule_MSGRENDERERS RegisterNamespace("MAIL:SUMM:INREPLYTO", 0, 2, tmplput_MAIL_SUMM_INREPLYTO, CTX_MAILSUM); RegisterNamespace("MAIL:BODY", 0, 2, tmplput_MAIL_BODY, CTX_MAILSUM); RegisterNamespace("MAIL:QUOTETEXT", 1, 2, tmplput_QUOTED_MAIL_BODY, CTX_NONE); + RegisterNamespace("MAIL:EDITTEXT", 1, 2, tmplput_EDIT_MAIL_BODY, CTX_NONE); RegisterConditional(HKEY("COND:MAIL:SUMM:RFCA"), 0, Conditional_MAIL_SUMM_RFCA, CTX_MAILSUM); RegisterConditional(HKEY("COND:MAIL:SUMM:CCCC"), 0, Conditional_MAIL_SUMM_CCCC, CTX_MAILSUM); RegisterConditional(HKEY("COND:MAIL:SUMM:UNREAD"), 0, Conditional_MAIL_SUMM_UNREAD, CTX_MAILSUM); @@ -1179,6 +1535,7 @@ InitModule_MSGRENDERERS /* mime renderers translate an attachment into webcit viewable html text */ RegisterMimeRenderer(HKEY("message/rfc822"), render_MAIL); + RegisterMimeRenderer(HKEY("text/vnote"), render_MIME_VNote); RegisterMimeRenderer(HKEY("text/x-vcard"), render_MIME_VCard); RegisterMimeRenderer(HKEY("text/vcard"), render_MIME_VCard); RegisterMimeRenderer(HKEY("text/calendar"), render_MIME_ICS); @@ -1217,3 +1574,30 @@ InitModule_MSGRENDERERS RegisterMsgHdr(HKEY("suff"), examine_suff, 0); RegisterMsgHdr(HKEY("path"), examine_path, 0); } + +void +ServerStartModule_MSGRENDERERS +(void) +{ + MsgHeaderHandler = NewHash(1, NULL); + MimeRenderHandler = NewHash(1, NULL); + ReadLoopHandler = NewHash(1, NULL); +} + +void +ServerShutdownModule_MSGRENDERERS +(void) +{ + DeleteHash(&MsgHeaderHandler); + DeleteHash(&MimeRenderHandler); + DeleteHash(&ReadLoopHandler); +} + + + +void +SessionDestroyModule_MSGRENDERERS +(wcsession *sess) +{ + DeleteHash(&sess->attachments); +}