/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*
*/
#include "common/formats/markdown.h"
#include "graphics/macgui/mactext.h"
namespace Graphics {
#define PR(x) ((x && x->data) ? Common::toPrintable(Common::String((const char *)(x)->data , (x)->size)).c_str() : "(null)")
struct MDState {
Common::List listNum;
uint32 linkr = 0, linkg = 0, linkb = 0;
};
void render_blockcode(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, const Common::SDDataBuffer *lang, void *opaque) {
if (!text)
return;
Common::String res = Common::String::format("\n\001\016+0001" "\001\016t%04x" "%s" "\001\016tffff" "\n\001\016-0001",
kMacFontMonaco, Common::String((const char *)text->data , text->size).c_str());
sd_bufput(ob, res.c_str(), res.size());
debug(1, "render_blockcode(%s, %s)", PR(text), PR(lang));
}
void render_blockquote(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text)
return;
warning("STUB: render_blockquote(%s)", PR(text));
}
void render_blockhtml(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text)
return;
warning("STUB: render_blockhtml(%s)", PR(text));
}
void render_header(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, int level, void *opaque) {
if (!text)
return;
debug(1, "render_header(%s)", PR(text));
Common::String res = Common::String::format("\001\016+00%01x0" "%s" "\001\016-00f0\n", level, Common::String((const char *)text->data , text->size).c_str());
sd_bufput(ob, res.c_str(), res.size());
}
void render_hrule(Common::SDDataBuffer *ob, void *opaque) {
warning("STUB: render_hrule()");
}
void render_list_start(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, int flags, void *opaque) {
MDState *mdstate = (MDState *)opaque;
mdstate->listNum.push_back(flags & MKD_LIST_ORDERED ? 1 : -1);
sd_bufput(ob, "\001\016+0001", 7);
debug(1, "render_list_start(%s, %d)", PR(text), flags);
}
void render_list(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, int flags, void *opaque) {
MDState *mdstate = (MDState *)opaque;
mdstate->listNum.pop_back();
sd_bufput(ob, text->data, text->size);
sd_bufput(ob, "\n\001\016-0001", 8);
debug(1, "render_list(%s, %d)", PR(text), flags);
}
void render_listitem(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, int flags, void *opaque) {
MDState *mdstate = (MDState *)opaque;
int listNum = mdstate->listNum.back();
Common::String prefix;
if (flags & MKD_LIST_ORDERED) {
prefix = Common::String::format("%d. ", listNum);
mdstate->listNum.back()++;
} else {
prefix = "* ";
}
Common::String res = Common::String::format("\001\016*%02x%s", prefix.size(), prefix.c_str());
sd_bufput(ob, res.c_str(), res.size());
sd_bufput(ob, prefix.c_str(), prefix.size());
sd_bufput(ob, text->data, text->size);
debug(1, "render_listitem(%s, %d)", PR(text), flags);
}
void render_paragraph(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text)
return;
debug(1, "render_paragraph(%s)", PR(text));
sd_bufput(ob, text->data, text->size);
sd_bufput(ob, "\n\n", 2);
}
void render_table(Common::SDDataBuffer *ob, const Common::SDDataBuffer *header, const Common::SDDataBuffer *body, void *opaque) {
if (!body)
return;
Common::String res = Common::String::format("\001\016Th" "%s" "\001\016Tb" "%s" "\001\016TB",
Common::String((const char *)header->data , header->size).c_str(), Common::String((const char *)body->data , body->size).c_str());
sd_bufput(ob, res.c_str(), res.size());
debug(1, "render_table(%s, %s)", PR(header), PR(body));
}
void render_table_row(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text)
return;
Common::String res = Common::String::format("\001\016Tr" "%s\n", Common::String((const char *)text->data , text->size).c_str());
sd_bufput(ob, res.c_str(), res.size());
debug(1, "render_table_row(%s)", PR(text));
}
void render_table_cell(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, int flags, void *opaque) {
if (!text)
return;
TextAlign align;
switch (flags) {
case Common::MKD_TABLE_ALIGN_R:
align = kTextAlignRight;
break;
case Common::MKD_TABLE_ALIGN_CENTER:
align = kTextAlignCenter;
break;
case Common::MKD_TABLE_ALIGN_L:
default:
align = kTextAlignLeft;
}
Common::String res = Common::String((const char *)text->data, text->size);
if (flags & Common::MKD_TABLE_HEADER)
res = Common::String::format("\001\016+%02x00" "%s" "\001\016-%02x00", kMacFontBold, res.c_str(), kMacFontBold);
res = Common::String::format("\001\016Tc%02x" "%s" "\001\016TC", align, res.c_str());
sd_bufput(ob, res.c_str(), res.size());
debug(1, "render_table_cell(%s), flags: %d", PR(text), flags);
}
int render_autolink(Common::SDDataBuffer *ob, const Common::SDDataBuffer *link, Common::MKDAutolink type, void *opaque) {
if (!link)
return 0;
warning("STUB: render_autolink(%s)", PR(link));
return 1;
}
int render_codespan(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text)
return 0;
Common::String res = Common::String::format("\001\016t%04x" "%s" "\001\016tffff",
kMacFontMonaco, Common::String((const char *)text->data , text->size).c_str());
sd_bufput(ob, res.c_str(), res.size());
debug(1, "render_codespan(%s)", PR(text));
return 1;
}
int render_double_emphasis(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text || !text->size)
return 0;
debug(1, "render_double_emphasis(%s)", PR(text));
Common::String res = Common::String::format("\001\016+%02x00" "%s" "\001\016-%02x00", kMacFontBold, Common::String((const char *)text->data , text->size).c_str(), kMacFontBold);
sd_bufput(ob, res.c_str(), res.size());
return 1;
}
int render_emphasis(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text || !text->size)
return 0;
debug(1, "render_emphasis(%s)", PR(text));
Common::String res = Common::String::format("\001\016+%02x00" "%s" "\001\016-%02x00", kMacFontItalic, Common::String((const char *)text->data , text->size).c_str(), kMacFontItalic);
sd_bufput(ob, res.c_str(), res.size());
return 1;
}
int render_image(Common::SDDataBuffer *ob, const Common::SDDataBuffer *link, const Common::SDDataBuffer *title, const Common::SDDataBuffer *alt, const Common::SDDataBuffer *ext, void *opaque) {
if (!link)
return 0;
Common::String res = Common::String::format("\001" "\016i%02x" "%02x%s",
80, (uint)link->size, Common::String((const char *)link->data, link->size).c_str());
if (alt) {
uint32 len = Common::U32String((const char *)alt->data, alt->size).size();
res += Common::String::format("%02x%s", len, Common::String((const char *)alt->data, alt->size).c_str());
} else {
res += "00";
}
if (title) {
uint32 len = Common::U32String((const char *)title->data, title->size).size();
res += Common::String::format("%02x%s", len, Common::String((const char *)title->data, title->size).c_str());
} else {
res += "00";
}
if (ext) {
uint32 len = Common::U32String((const char *)ext->data, ext->size).size();
res += Common::String::format("%02x%s", len, Common::String((const char *)ext->data, ext->size).c_str());
} else {
res += "00";
}
res += "\n";
sd_bufput(ob, res.c_str(), res.size());
debug(1, "render_image(%s, %s, %s, %s)", PR(link), PR(title), PR(alt), PR(ext));
return 1;
}
int render_linebreak(Common::SDDataBuffer *ob, void *opaque) {
debug(1, "render_linebreak()");
sd_bufput(ob, "\n", 1);
return 1;
}
int render_link(Common::SDDataBuffer *ob, const Common::SDDataBuffer *link, const Common::SDDataBuffer *title, const Common::SDDataBuffer *content, void *opaque) {
if (!link)
return 0;
MDState *mdstate = (MDState *)opaque;
const Common::SDDataBuffer *text = content ? content : link;
uint32 linklen = Common::U32String((const char *)link->data, link->size).size();
Common::String res = Common::String::format("\001" "\016+%02x00" "\001\016[%04x%04x%04x"
"\001\016l%02x%s" "%s" "\001\016l00" "\001\016]" "\001\016-%02x00", kMacFontUnderline,
mdstate->linkr, mdstate->linkg, mdstate->linkb,
linklen, Common::String((const char *)link->data , link->size).c_str(),
Common::String((const char *)text->data , text->size).c_str(), kMacFontUnderline);
sd_bufput(ob, res.c_str(), res.size());
debug(1, "render_link(%s, %s, %s)", PR(link), PR(title), PR(content));
return 1;
}
int render_raw_html_tag(Common::SDDataBuffer *ob, const Common::SDDataBuffer *tag, void *opaque) {
if (!tag)
return 0;
sd_bufput(ob, tag->data, tag->size);
debug(1, "render_raw_html_tag(%s)", PR(tag));
return 1;
}
int render_triple_emphasis(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text)
return 0;
warning("STUB: render_triple_emphasis(%s)", PR(text));
return 1;
}
int render_strikethrough(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text)
return 0;
warning("STUB: render_strikethrough(%s)", PR(text));
return 1;
}
int render_superscript(Common::SDDataBuffer *ob, const Common::SDDataBuffer *text, void *opaque) {
if (!text)
return 0;
warning("STUB: render_superscript(%s)", PR(text));
return 1;
}
void MacText::setMarkdownText(const Common::U32String &str) {
const Common::SDCallbacks cb = {
/* block level callbacks - NULL skips the block */
render_blockcode,
render_blockquote,
render_blockhtml,
render_header,
render_hrule,
render_list_start,
render_list,
render_listitem,
render_paragraph,
render_table,
render_table_row,
render_table_cell,
/* span level callbacks - NULL or return 0 prints the span verbatim */
render_autolink,
render_codespan,
render_double_emphasis,
render_emphasis,
render_image,
render_linebreak,
render_link,
render_raw_html_tag,
render_triple_emphasis,
render_strikethrough,
render_superscript,
/* low level callbacks - NULL copies input directly into the output */
NULL,
NULL,
/* header and footer */
NULL,
NULL,
};
Common::String input = str.encode(); // Encode to UTF8
MDState mdState;
// Set link color to blue
mdState.linkr = 0;
mdState.linkg = 0;
mdState.linkb = 0xff;
Common::SDMarkdown md(Common::MKDEXT_TABLES, 16, &cb, &mdState);
Common::String rendered = md.render((const byte *)input.c_str(), input.size());
setText(rendered);
}
} // End of namespace Graphics