#include "www.h" #include #include #define IMAGECLASS WWW #define IMAGEFILE #include #define TFILE #include #define LLOG(x) // LOG(x) #ifdef PLATFORM_WIN32 String rootdir = "d:\\devweb\\upp.src"; String uppbox = rootdir + "uppbox"; String uppsrc = rootdir + "uppsrc"; String reference = rootdir + "reference"; String examples = rootdir + "examples"; String targetdir = "u:\\devweb\\uppwww"; String pdfdir = "u:\\devweb\\pdf"; #else #ifdef _DEBUG String rootdir = GetHomeDirFile("upp.src"); String uppbox = rootdir + "uppbox"; String uppsrc = rootdir + "uppsrc"; String reference = rootdir + "reference"; String examples = rootdir + "examples"; String targetdir = GetHomeDirFile("www"); String diffdir = GetHomeDirFile("wwwd"); String pdfdir = GetHomeDirFile("www"); #else String rootdir = "/root/upp.src"; String uppbox = rootdir + "uppbox"; String uppsrc = rootdir + "uppsrc"; String reference = rootdir + "reference"; String examples = rootdir + "examples"; String targetdir = "/var/www"; String diffdir = "/root/wwwupp"; String pdfdir = "/var/www"; #endif #endif String bazaar; bool outPdf; bool outHtml; bool doGit; String GetRcFile(const char *s) { String f = GetDataFile(s); if(FileExists(f)) return f; return GetHomeDirFile("upp.src/uppbox/uppweb/" + AsString(s)); } bool ContainsAt(const String &source, const String &pattern, int pos) { return pos >= 0 && pos + pattern.GetLength() <= source.GetLength() && 0 == memcmp(source.Begin() + pos, pattern.Begin(), pattern.GetLength()); } bool StartsWith(const String &source, const String &pattern) { return ContainsAt(source, pattern, 0); } bool EndsWith(const String &source, const String &pattern) { return ContainsAt(source, pattern, source.GetLength() - pattern.GetLength()); } String ImgName(int q) { return Sprintf("%di.png", q); } typedef Image (*ImageFn)(); namespace Upp { template<> hash_t GetHashValue(const ::ImageFn& fn) { return (unsigned)(uintptr_t) fn; } }; String GetImageSrc(ImageFn img) { static Index il; int q = il.Find(img); if(il.Find(img) < 0) { q = il.GetCount(); il.Add(img); PNGEncoder png; png.SaveFile(AppendFileName(targetdir, ImgName(q)),(*img)()); } return ImgName(q); } Htmls Wimg(ImageFn img) { return HtmlImg(GetImageSrc(img)).Border(0); } Htmls RoundFrame(Htmls data, String border, Color bg) { return HtmlPackedTable().BgColor(bg).Width(-100) .Attr("style", "border-style: solid; border-width: 1px; border-color: #" + border + ";") / HtmlLine() / data; } HtmlTag BoxWidth(int width) { return HtmlPackedTable().Width(width) / HtmlLine(); } Htmls BarSection(const char *txt) { Htmls bar; bar.Br(); bar << HtmlPackedTable().BgColor(WhiteGray).Width(-100) / (HtmlRow() / ( HtmlCell().Width(10) / " " + HtmlCell() / HtmlArial(8) / HtmlBold() / txt + HtmlCell().Width(10) / " " )); return bar; } Htmls BarCaption(const char *text) { return HtmlLine() .Width(-100) .VCenter() .Style("background-image: url('" + GetImageSrc(WWW::Caption) + "'); " "border: 0px solid black;" "padding-left:12px; padding-right:0px; " "padding-top:1px; padding-bottom:1px;" "color:#FFFFFF;") / HtmlBold() / HtmlArial(8) / text; } Htmls BarCaptionLang(const char *text) { return HtmlLine() .Width(20) .VCenter() .Style("background-image: url('" + GetImageSrc(WWW::News2) + "'); " "border: 0px solid black;" "padding-left:12px; padding-right:0px; " "padding-top:1px; padding-bottom:1px;" "color:#FFFFFF;") / HtmlBold() / HtmlArial(8) / text; } Htmls BarItem(Htmls content, const char *style) { String bgStyle = "background-image: url('" + GetImageSrc(WWW::Button) + "'); "; return HtmlLine().Width(-100).VCenter().Style(bgStyle + style) / content; } Htmls BarLink(const char *link, const char *text, bool nxt = true) { String style = "border: 0px solid black; " "padding-left:12px; padding-right:0px; " "padding-top:6px; padding-bottom:6px;"; if(nxt) style += " border-top: 1px solid #6E89AE;"; return BarItem( HtmlLink(link).Class("l1") / text, style ); } Htmls SearchBar(const char *domain) { Htmls form = HtmlForm("https://www.google.com/search", false, "GET") .Style("margin:0px;padding:0px;") / ( HtmlHidden("ie", "UTF-8") + HtmlHidden("oe", "UTF-8") + ~HtmlEdit("q", 15).Attr("placeholder", t_("Site search")) + ~HtmlHidden("sitesearch", domain) ); Htmls content; content << HtmlPackedTable().Width(-100) / HtmlRow() / ( HtmlCell() / Wimg(WWW::google) + HtmlCell() / form ); String style = "border: 0px solid black; " "padding-left:6px; padding-right:0px; " "padding-top:4px; padding-bottom:4px;" "border-top: 1px solid #6E89AE;"; return BarItem(content, style); } HtmlTag HtmlPadding(int p) { return HtmlPackedTable().Width(-100) / HtmlLine(); } VectorMap escape; String QtfAsHtml(const char *qtf, Index& css, const VectorMap& links, const VectorMap& labels, const String& outdir, const String& fn = Null) { return EncodeHtml(ParseQTF(qtf), css, links, labels, outdir, fn, Zoom(8, 40), escape, 40); } String GetText(const char *s) { return GetTopic(s).text; } String ChangeTopicLanguage(const String &topic, int lang) { int pos = topic.ReverseFind('$'); if (pos < 0) return ""; String langtxt = ToLower(LNGAsText(lang)); return topic.Left(pos+1) + langtxt + topic.Mid(pos+1+langtxt.GetCount()); } String GetTopicLanguage(const String &topic) { int pos = topic.ReverseFind('$'); if (pos < 0) return ""; return topic.Mid(pos+1, 5); } String FormatDateRFC822(const Time& t) { int tz = int((GetSysTime()-GetUtcTime())/60); return Format("%s, %d %Mon %d %02d:%02d:%02d %s%04d", DayName(DayOfWeek(t)).Left(3), t.day,t.month,t.year, t.hour,t.minute,t.second, tz>0?"+":"",tz/60*100+(tz+1440)%60); } ArrayMap tt; Vector ttId; Vector ttFullIds; String Www(const char *topic, int lang, String topicLocation = "topic://uppweb/www/") { String strLang = ToLower(LNGAsText(lang)); String www = GatherTopics(tt, ttFullIds, String().Cat() << topicLocation << topic << "$" << strLang, ""); if (www != "index.html") return www; return GatherTopics(tt, ttFullIds, String().Cat() << topicLocation << topic << "$" << "en-us", ""); } struct Title { String title, link; Title(String title, String link) : title(TrimBoth(title)), link(link) {}; bool operator < (const Title& other) const { const char *val = title.Begin(); const char *valOther = other.title.Begin(); for(;; val++, valOther++) { String str, strOther; str.Clear(); strOther.Clear(); for (; *val != '\0' && *val != '.' && *val != ' '; ++val) str.Cat(*val); for (; *valOther != '\0' && *valOther != '.' && *valOther != ' '; ++valOther) strOther.Cat(*valOther); int num = ScanInt(str); int numOther = ScanInt(strOther); if (IsNull(num) && IsNull(numOther)) return str < strOther; if (IsNull(num)) return true; if (IsNull(numOther)) return false; if (num != numOther) return num < numOther; } return title < other.title; } }; String FolderLinks(String package, String group, int lang, int level, String parentIds) { String qtf; Array titles; FindFile ff(AppendFileName(AppendFileName(AppendFileName(uppsrc, package), group + ".tpp"), "*.tpp")); while(ff) { if(ff.IsFile()) { if (ff.GetName().Find("en-us") >= 0) { String title; String tl = "topic://" + package + '/' + group + '/' + GetFileTitle(ff.GetName()); GatherTopics(tt, ttFullIds, tl, title, parentIds); titles.Add(Title(DeQtf(Nvl(title, tl)), tl)); //qtf << "________[^" << tl << "^ " << DeQtf(Nvl(title, tl)) << "]&"; } } ff.Next(); } Sort(titles); for (int i = 0; i < titles.GetCount(); ++i) qtf << "________[^" << titles[i].link << "^ " << titles[i].title << "]&"; return qtf; } void AddFiles(String& qtf, const String& dir, const char* ext, bool& b) { FindFile ff(AppendFileName(dir, "*." + String(ext))); while(ff) { qtf << "[A4* " << DeQtf(ff.GetName()) << "]&&" << CppAsQtf(LoadFile(AppendFileName(dir, ff.GetName()))) << "&&&"; ff.Next(); b = true; } } struct Isort { bool operator()(const String& a, const String& b) const { return ToUpper(a) < ToUpper(b); } }; String MakeExamples(const char *dir, const char *www, int language, String parentIds) { String ttxt; FindFile ff(AppendFileName(dir, "*.*")); ttxt << "{{1:3 "; bool next = false; Vector<String> ls; while(ff) { if(ff.IsFolder()) ls.Add(ff.GetName()); ff.Next(); } Sort(ls, Isort()); for(int i = 0; i < ls.GetCount(); i++) { String name = ls[i]; String link = String().Cat() << www << '$' << name << "$" << ToLower(LNGAsText(language)) << ".html"; Topic& topic = tt.Add(link); topic.title = name; int q = tt.GetCount() - 1; String fullIds = parentIds + "/" + FormatInt(q); ttFullIds.Add(fullIds); String fn = AppendFileName( AppendFileName( AppendFileName(uppbox, "uppweb"), String(www) + ".tpp" ), topic.title + "_" + ToLower(LNGAsText(language)) + ".tpp" ); String h = ReadTopic(LoadFile(fn)).text; Package p; p.Load(AppendFileName(AppendFileName(dir, name), name + ".upp")); topic.text << "[R6* " << name << "]&&" << DeQtf(p.description) << "&"; if(h.GetCount()) topic.text << h; topic.text << "[A2<l0r0 &&"; String d = AppendFileName(dir, name); bool b = false; AddFiles(topic.text, d, "h", b); AddFiles(topic.text, d, "hpp", b); AddFiles(topic.text, d, "cpp", b); AddFiles(topic.text, d, "icpp", b); AddFiles(topic.text, d, "java", b); AddFiles(topic.text, d, "witz", b); AddFiles(topic.text, d, "usc", b); AddFiles(topic.text, d, "lay", b); AddFiles(topic.text, d, "key", b); AddFiles(topic.text, d, "brc", b); AddFiles(topic.text, d, "sch", b); AddFiles(topic.text, d, "xml", b); if(b) { if(next) ttxt << "\n::^ "; ttxt << "[^" << link << "^* " << DeQtf(topic.title) << "]::^ " << DeQtf(p.description); next = true; } } ttxt << "}}&&"; return ttxt; } class HeaderNumber { public: HeaderNumber(const String& prefix = {}) : m_prefix(prefix) { } bool IsFirst() const { return m_number == 1; } int GetNumber() const { return m_number; } String GenNext() { return m_prefix + IntStr(m_number++); } private: String m_prefix; int m_number = 1; }; class PackageDocsGenerator final { public: PackageDocsGenerator() : m_hnumber("4.") {} const Vector<String>& GetHeaders() { return m_headers; } void Generate(Index<String>& x, String& qtf, const char* folder, int lang, int level, String parent) { if(x.Find(folder) >= 0) return; x.Add(folder); String srcdoc = FolderLinks(folder, "srcdoc", lang, level, parent); String src = FolderLinks(folder, "src", lang, level, parent); Package p; p.Load(AppendFileName(uppsrc, AppendFileName(folder, GetFileName(folder) + ".upp"))); if(srcdoc.GetLength() || src.GetLength()) { bool first = m_hnumber.IsFirst(); String ref = ":4`_" + IntStr(m_hnumber.GetNumber()) + ":"; m_headers.Add(m_hnumber.GenNext() + " " + folder); qtf << (first ? "" : "[s0; &&]\n") << "[s19;" << ref << " " << m_headers.Top() << " 📦&]"; if(!IsNull(p.description)) qtf << "[2 " << p.description << "&]"; if(srcdoc.GetCount()) { qtf << "&[3/* " + Format(t_("Using %s"), folder) << "&]"; qtf << srcdoc; } if(src.GetCount()) { qtf << "&[3/* " << Format(t_("%s reference"), folder) << "&]"; qtf << src; } qtf << "\n"; } } private: HeaderNumber m_hnumber; Vector<String> m_headers; }; int CharFilterLbl(int c) { return IsAlNum(c) ? c : '.'; } void QtfAsPdf(PdfDraw &pdf, const char *qtf) { RichText txt = ParseQTF(qtf); Size page = Size(3968, 6074); UPP::Print(pdf, txt, page); } VectorMap<String, String> links; VectorMap<String, String> labels; Htmls header, lastUpdate; Array <Htmls> bar; Array <int> languages; int GetLinkLanguage(const String &link) { int pos = link.ReverseFind('$'); if (pos < 0) return 0; int lang = LNGFromText(ToUpper(link.Mid(pos+1))); for (int i = 0; i < languages.GetCount(); ++i) { if (languages[i] == lang) return i; } return 0; } String TrimDescription(String description) { description.Replace("\"", ""); description.Replace(":", "."); constexpr int DESCRIPTION_SIZE_LIMIT = 158; if (description.GetLength() <= DESCRIPTION_SIZE_LIMIT) { return description; } description.Trim(DESCRIPTION_SIZE_LIMIT); int trim_idx = -1; for (int i = description.GetCount() - 1; i >= 0; i--) { if (description[i] != '.' && description[i] != '?') { continue; } trim_idx = i + 1; break; } if (trim_idx == -1 || trim_idx == 0) { return description; } description.Trim(trim_idx); return description; } String FindPageDescription(const RichText& qtf) { String description; Uuid style_id = qtf.GetStyleId("text"); if (!style_id.IsNullInstance()) { RichStyle style = qtf.GetStyle(style_id); for (int i = 0; i < qtf.GetPartCount(); i++) { if (!qtf.IsPara(i)) { continue; } RichPara para = qtf.Get(i); if (style_id == qtf.GetParaStyle(i)) { description = para.GetText().ToString(); break; } } } return TrimDescription(description); } void ExportPage(int i) { Index<String> css; String path = links.GetKey(i); RLOG("Exporting " << path); int ilang = GetLinkLanguage(path); SetLanguage(languages[ilang]); String text = GetText(path); RichText qtf = ParseQTF(tt[i].text); int h = qtf.GetHeight(1000); int isvn = svndata.Find(tt.GetKey(i)); String qtflangs; String googleFile; if (isvn > -1) { String txt = String("[2 ") + t_("Last edit by %s on %s") + ".]"; qtflangs += Format(txt, svndata[isvn].author, Format(Date(svndata[isvn].time))); googleFile = svndata[isvn].fullPath; if (googleFile.GetCount() > rootdir.GetCount()) googleFile = UnixPath(googleFile.Mid(rootdir.GetCount())); else googleFile = ""; } String strlang; String jslang; Array <String> arrLangs; for (int il = 0; il < languages.GetCount(); ++il) { String topic = ChangeTopicLanguage(path, languages[il]); int itopic; if ((itopic = tt.Find(topic)) >= 0) { if (tt[itopic].title.Find(" (translated)") < 0) { if (il != ilang) { if (!strlang.IsEmpty()) strlang << ", "; arrLangs.Add("[^" + links[itopic] + "^ [2 " + ToLower(GetNativeLangName(languages[il])) + "]]"); } jslang+="'"+ links[itopic] +"':'" + GetNativeLangName(languages[il]) + "',"; } } } if(!jslang.IsEmpty()) jslang.Trim(jslang.GetCount()-1); if (arrLangs.GetCount() > 0) { for (int i = 0; i < arrLangs.GetCount(); ++i) { if (i == arrLangs.GetCount()-1 && i != 0) strlang << String(" ") + t_("and") + " "; else if (i > 0) strlang << ", "; strlang << arrLangs[i]; } } // if (!strlang.IsEmpty()) // qtflangs += Format(String("[2 ") + t_("This page is also in %s") + ".]", strlang); if (tt[i].title.Find("How to contribute. Web page") < 0) { String help = "topic://uppweb/www/contribweb$" + ToLower(LNGAsText(languages[ilang])); qtflangs += " " + String("[^") + help + "^ [<A2 " + t_("Do you want to contribute?") + "]]"; if (googleFile != "") qtflangs += ". [^https`:`/`/raw`.githubusercontent`.com`/ultimatepp`/mirror`/master" + DeQtf(UrlEncode(googleFile)) + "^/1 " + "T`+`+" + "]"; } String langs = QtfAsHtml(qtflangs, css, links, labels, targetdir, links[i]); String page = tt[i]; Array<String> htmlrep; int posB = 0; while (true) { posB = page.Find("[IHTMLTEXT", posB); if (posB < 0) break; int posBB = posB + (int)strlen("[IHTMLTEXT"); int pos0 = page.ReverseFind("[", posB-1); int posE = page.Find(";2", posBB); String html0 = page.Mid(posBB, posE - posBB); html0.Replace("`", ""); htmlrep.Add(html0); int posEE = page.Find("]&]", posE) + (int)strlen("]&]"); page = page.Left(pos0) + "QTFHTMLTEXT" + FormatInt(htmlrep.GetCount()-1) + page.Mid(posEE+1); } page = QtfAsHtml(page, css, links, labels, targetdir, links[i]); for (int iHtml = 0; iHtml < htmlrep.GetCount(); ++iHtml) page.Replace(String("QTFHTMLTEXT") + FormatInt(iHtml), htmlrep[iHtml]); Color paper = SWhite; if(path == "topic://uppweb/www/download$en-us") page << LoadFile(GetRcFile("adsense3.txt")); /* if(path == "topic://uppweb/www/index$en-us") { for(int q = 0; q < news.GetCount(); q++) { String n = GetText("uppweb/www_news/" + news[q]); String h = news[q]; int i = h.Find('$'); if(i >= 0) h = h.Mid(0, i); if(h.GetLength() == 8) h = h.Mid(0, 4) + '-' + h.Mid(4, 2) + '-' + h.Mid(6, 2); page << "<br>"; page << "<div style='font-family:sans-serif; font-weight: bold; " "font-height: 12px; color: White; background: #2020d0'>  " << h << "</div><br>"; page << QtfAsHtml(n, css, links, targetdir, FormatIntAlpha(q) + "_n"); page << "<br>"; } }*/ Color bg = Color(210, 217, 210); Htmls footer; footer << HtmlTable().Border(0).Width(-100) / HtmlLine() + ~RoundFrame(HtmlPadding(8) / langs , "6E89AE;padding: 10px;", White); Htmls html; html << HtmlPackedTable().Width(-100) / HtmlLine().ColSpan(3) / header + HtmlPackedTable().Width(-100) / ( HtmlLine().ColSpan(3).BgColor(bg).Height(6) / "" + HtmlRow() / ( HtmlTCell().Center() / BoxWidth(160).Center() / ( bar[GetLinkLanguage(path)] + "<br>" + LoadFile(GetRcFile("facebook.txt")) + "<br><br>" + // "<p align=\"center\">" + LoadFile(GetRcFile("adsense2.txt")) + "<br><br>" + LoadFile(GetRcFile("adlinks.txt")) + (h > 25000 ? "<br><br>" + LoadFile(GetRcFile("adsense2.txt")) : "") + "<br><br><br>" + // LoadFile(GetRcFile("referral.txt")) + // LoadFile(GetRcFile("referral2.txt")) + // LoadFile(GetRcFile("donations.txt")) + // "<br><br>" + // amazon[i % amazon.GetCount()] + "<br><br><br>" + ~(HtmlLink("https://sourceforge.net/projects/upp/") / HtmlImg("https://sourceforge.net/sflogo.php?group_id=93970&type=2", "SourceForge.net Logo").Border(0).Width(125).Height(37)) + "<br><br>" + ~(HtmlLink("https://flathub.org/en/apps/org.ultimatepp.TheIDE") / HtmlImg(GetImageSrc(WWW::Flathub), "SourceForge.net Logo").Border(0).Width(125).Height(37)) + "<br><br>" + "<div style=\"background-color:#ffffff;width:125;height:35\">" + ~(HtmlLink("https://github.com/ultimatepp") / HtmlImg(GetImageSrc(WWW::GitHub), "GitHub Logo").Border(0).Width(125).Height(35)) + "</div>" + "<br>" + "<div style=\"background-color:#ffffff;width:125;height:35\">" + "<div style=\"height:5\"></div>" + ~(HtmlLink("https://discord.gg/8XzqQzXZzb") / HtmlImg("https://assets-global.website-files.com/6257adef93867e50d84d30e2/636e0b5061df29d55a92d945_full_logo_blurple_RGB.svg", "Discord Logo").Border(0).Width(125).Height(25)) + "</div>" + "<br>" + (links[i] == "index.html" ? lastUpdate : Htmls()) ) + HtmlTCell().BgColor(bg) / BoxWidth(6) / "" + HtmlTCell().Width(-100).BgColor(bg) / ( RoundFrame(HtmlPadding(8) / page , "6E89AE;padding: 10px;", White) + footer ) ) ) + Htmls("<script type=\"text/javascript\">var a={"+jslang+"};" "var p=window.location.pathname.split(\"/\");p=p[p.length-1];" "var s='<select id=\"lang\" onchange=\"window.location=document.getElementById(\\'lang\\').value;\">';" "for(l in a){if(p==l){d=\" selected=1\"}else{d=\"\"}s+='<option value=\"'+l+'\"'+d+'>'+a[l]+'</option>'}" "s+='</select>';" "var d=document.createElement('div');d.innerHTML=s;" "var c=document.getElementById('langs');c.replaceChild(d,c.firstChild);" "document.getElementById('langbox').style.display='block';</script>"); String topicTitle = tt.GetKey(i); String pageTitle = tt[i].title; if(IsNull(pageTitle)) pageTitle = "U++"; else if(StartsWith(topicTitle, "examples")) pageTitle = "Demos / " + pageTitle; else if(StartsWith(topicTitle, "reference")) pageTitle = "Examples / " + pageTitle; bool is_index = path == "topic://uppweb/www/index$en-us"; if(pageTitle != "U++" && !is_index) pageTitle << " :: U++"; auto htmlHeader = HtmlHeader() .Title(pageTitle) .Description(FindPageDescription(qtf)) .Css(AsCss(css) + "a.l1 { text-decoration:none; font-size: 8pt; font-family: sans-serif; " "font-weight: normal; }\n" "a.l1:link { color:#000000; }\n" "a.l1:visited { color:#000080; }\n" "a.l1:hover { color:#9933CC; }\n" "a.l1:active { color:#000000; }\n" "a.l2 { text-decoration:none; font-size: 12pt; font-family: sans-serif; " "font-variant: small-caps; }\n" "a.l2:link { color:#0066FF; }\n" "a.l2:visited { color:#FF6600; }\n" "a.l2:hover { color:#BC0624; }\n" "a.l2:active { color:#BC0024; }\n") .Other( "<META NAME=\"keywords\" " "CONTENT=\"" "framework, toolkit, widget, c++, visual, studio, dev-cpp, builder, ide, class, component," "wxwidgets, qt, rapid, application, development, rad, mfc, linux, macos, gui, sdl, directx, desktop" "\">" "<META name=\"robots\" content=\"index,follow\">\n" "<LINK rel=\"alternate\" type=\"application/rss+xml\" title=\"SVN changes\" href=\"svnchanges.xml\">\n" "<LINK rel=\"shortcut icon\" type=\"image/png\" href=\"favicon.png\">\n") .Create(); Htmls content = //"<!DOCTYPE html>" + "<!DOCTYPE html PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\">\n" + htmlHeader .BgColor(bg) .Alink(Red).Link(Black).Vlink(Blue) / html; if(!SaveFile(AppendFileName(targetdir, links[i]), content)) { RLOG("SaveFile failed!"); abort(); } RLOG("Exported page " << AppendFileName(targetdir, links[i])); } struct DownloadItem { Time tm; String fn; String path; int64 len; }; String Downloads() { String r; r << "{{1:2:2^@L " "Date:: " "[^app$ide$install_win32_en-us.html^ U`+`+ for Windows (with CLANG)]:: " "[^app$ide$install_posix_en-us.html^ U`+`+ for Linux/MacOS/BSD/Solaris]"; FindFile ff(AppendFileName(targetdir, "downloads/*.*")); SortedArrayMap<Date, Array<DownloadItem>> downs; while(ff) { if(ff.IsFile()) { DownloadItem down; down.tm = ff.GetLastWriteTime(); down.fn = ff.GetName(); down.path = ff.GetPath(); down.len = ff.GetLength(); Date date = down.tm; downs.GetAdd(date).Add(down); } ff.Next(); } auto FindDown = [](Array<DownloadItem>& a, const char* arch) -> String { String r; for(int i = 0; i < a.GetCount(); i++) { if(a[i].fn.Find(arch) >= 0) { r << FormatTime(a[i].tm, "hh:mm") << " [^downloads/" << a[i].fn << "^ \1" << a[i].fn << "\1]" << " (" << (a[i].len >> 20) << " MB)"; break; } } return r; }; Date d = GetSysDate(); for(int i = 0; i < 14; i++, d--) { Array<DownloadItem>& a = downs.GetAdd(d); r << "::@W " << FormatDate(d, "YYYY-MM-DD") << ":: " << FindDown(a, "win") << ":: " << FindDown(a, "posix"); } r << "}}"; for(int i = 14; i < downs.GetCount(); i++) { if(downs.GetKey(i) < d) for(int j = 0; j < downs[i].GetCount(); j++) DeleteFile(downs[i][j].path); } return r; } struct ProgramData { String rootdir; String targetdir; String diffdir; String pdfdir; bool ftpUpload; bool outPdf; bool outHtml; bool doGit; void Xmlize(XmlIO xml) { xml ("rootdir", rootdir) ("targetdir", targetdir) ("diffdir", diffdir) ("pdfdir", pdfdir) ("ftpUpload", ftpUpload) ("outPdf", outPdf) ("doGit", doGit) ("outHtml", outHtml) ; } }; CONSOLE_APP_MAIN { StdLogSetup(LOG_COUT|LOG_FILE); #ifdef PLATFORM_POSIX rootdir = GetHomeDirFile("upp.src"); targetdir = GetHomeDirFile("uppwww"); diffdir = GetHomeDirFile("wwwupp"); pdfdir = GetHomeDirFile("pdf"); #endif outPdf = true; outHtml = true; doGit = true; RLOG("--- uppweb started at " << GetSysTime()); ProgramData data; String configFile = GetHomeDirFile("uppweb.xml"); const Vector<String>& cmd = CommandLine(); if(cmd.GetCount() && FileExists(cmd[0])) configFile = cmd[0]; Cout() << "ConfigFile: " << configFile << "\n"; bool cfgloaded = false; if (FileExists(configFile)) { if (LoadFromXMLFile(data, configFile)) { rootdir = data.rootdir; targetdir = data.targetdir; // diffdir = data.diffdir; pdfdir = data.pdfdir; // ftpupload = data.ftpUpload; outPdf = data.outPdf; outHtml = data.outHtml; doGit = data.doGit; cfgloaded = true; } } if (!cfgloaded) { data.rootdir = rootdir; data.targetdir = targetdir; // data.diffdir = diffdir; data.pdfdir = pdfdir; // data.ftpUpload = ftpupload; data.outPdf = outPdf; data.outHtml = outHtml; data.doGit = doGit; StoreAsXMLFile(data, NULL, configFile); } Cout() << "RootDir: " << rootdir << "\n"; Cout() << "OutHtml: " << outHtml << "\n"; Cout() << "TargetDir: " << targetdir << "\n"; Cout() << "OutPdf: " << outPdf << "\n"; Cout() << "PdfDir: " << pdfdir << "\n"; #ifdef _DEBUG outPdf = false; #endif String downloads = Downloads(); if (!DirectoryExists(rootdir)) { Cout() << ("Directory " + DeQtf(rootdir) + " does not exist\n"); return; } uppbox = AppendFileName(rootdir, "uppbox"); uppsrc = AppendFileName(rootdir, "uppsrc"); reference = AppendFileName(rootdir, "reference"); examples = AppendFileName(rootdir, "examples"); bazaar = AppendFileName(rootdir, "bazaar"); Cout() << "Loading www.tpp\n"; InitWwwTpp(); languages.Add(LNG_('E','N','U','S')); // en-us has to be the first one #if 0 #ifndef _DEBUG // too slow to have them all while developing languages.Add(LNG_('C','A','E','S')); languages.Add(LNG_('C','S','C','Z')); languages.Add(LNG_('D','E','D','E')); languages.Add(LNG_('E','S','E','S')); languages.Add(LNG_('E','U','E','S')); languages.Add(LNG_('F','R','F','R')); languages.Add(LNG_('R','O','R','O')); languages.Add(LNG_('R','U','R','U')); languages.Add(LNG_('Z','H','C','N')); languages.Add(LNG_('Z','H','T','W')); #endif #endif if (outHtml) RealizeDirectory(targetdir); if (outPdf) RealizeDirectory(pdfdir); Cout() << "Gather ref links " << uppsrc << "\n"; GatherRefLinks(uppsrc); Cout() << "Gather ref links " << AppendFileName(rootdir, "bazaar") << "\n"; GatherRefLinks(AppendFileName(rootdir, "bazaar")); if (outHtml) SaveFile(AppendFileName(targetdir, "sdj.gif"), LoadFile(GetRcFile("sdj.gif"))); String release = "17045"; escape.Add("RELEASE", release); escape.Add("RELEASET", release); escape.Add("UPDATETIME", Format("%`", GetUtcTime())); if (doGit) { Cout() << "Processing git\n"; #ifdef bmGIT_REVCOUNT auto version = AsString(atoi(bmGIT_REVCOUNT) + 2270); escape.Add("LATESTGIT", version); #endif } escape.Add("PAYPAL", LoadFile(GetRcFile("donations.txt"))); header = HtmlPackedTable() .Width(-100) .BgColor(White) .Attr("style", "border: 1px solid #6E89AE;" "padding-left: 10px;padding-right: 0px;padding-top: 0px;padding-bottom: 0px;") / HtmlRow() / ( HtmlCell() / HtmlLink("index.html") / Wimg(WWW::Logo6) + HtmlCell().Right().Bottom() .Style("padding-bottom: 5px; " "background-image: url('" + GetImageSrc(WWW::HB) + "')") / HtmlArial(14) / (LoadFile(GetRcFile("adsense.txt")) + "  "/* + "<br>..harnessing the real power of C++  "*/) ); bar.SetCount(languages.GetCount()); int currentLang = GetCurrentLanguage(); int di; for (int i = 0; i < languages.GetCount(); ++i) { Cout() << "Language " << LNGAsText(languages[i]); Htmls bi, bex, bdoc, bcom, bcon, bsearch, blang; int lang = languages[i]; SetLanguage(lang); Www("index", lang); Www("contribweb", lang); // bi << BarLink("index.html", "Home", false); bi << BarLink(Www("overview", lang), t_("Overview"), false); bi << BarLink(Www("examples", lang), t_("Examples")); { IGNORE_RESULT(Www("reference", lang)); int ri = tt.Find("topic://uppweb/www/reference$" + ToLower(LNGAsText(lang))); int di = tt.Find("topic://uppweb/www/examples$" + ToLower(LNGAsText(lang))); tt[di].text << MakeExamples(examples, "examples", lang, String("/") + FormatInt(di)); tt[di].text << tt[ri].text << '\n'; tt[di].text << MakeExamples(reference, "reference", lang, String("/") + FormatInt(di)); } bi << BarLink(Www("ss", lang), t_("Screenshots")); bi << BarLink(Www("comparison", lang), t_("Comparisons")); bi << BarLink(Www("apps", lang), t_("Applications")); bi << BarLink(Www("download", lang), t_("Download")); bi << BarLink(Www("documentation", lang), t_("Documentation")); { int di = tt.Find("topic://uppweb/www/documentation$" + ToLower(LNGAsText(lang))); if (di >= 0) { Index<String> x; x.Clear(); String qtf; PackageDocsGenerator dgen; dgen.Generate(x, qtf, "Core", lang, 2, String("/") + FormatInt(di) + "/[Core]"); dgen.Generate(x, qtf, "Core/POP3", lang, 2, String("/") + FormatInt(di) + "/[Core/POP3]"); dgen.Generate(x, qtf, "Core/SMTP", lang, 2, String("/") + FormatInt(di) + "/[Core/SMTP]"); dgen.Generate(x, qtf, "Core/SSH", lang, 2, String("/") + FormatInt(di) + "/[Core/SSH]"); dgen.Generate(x, qtf, "Draw", lang, 2, String("/") + FormatInt(di) + "/[Draw]"); dgen.Generate(x, qtf, "CtrlCore", lang, 2, String("/") + FormatInt(di) + "/[CtrlCore]"); dgen.Generate(x, qtf, "CtrlLib", lang, 2, String("/") + FormatInt(di) + "/[CtrlLib]"); dgen.Generate(x, qtf, "RichText", lang, 2, String("/") + FormatInt(di) + "/[RichText]"); dgen.Generate(x, qtf, "RichEdit", lang, 2, String("/") + FormatInt(di) + "/[RichEdit]"); dgen.Generate(x, qtf, "Sql", lang, 2, String("/") + FormatInt(di) + "/[Sql]"); String d = AppendFileName(uppsrc, "*.*"); String p; for(int pass = 0; pass < 2; pass++) { FindFile ff(d); Array <String> folders; folders.Clear(); while(ff) { if(ff.IsFolder() && ff.GetName() != "Web") folders.Add(p + ff.GetName()); ff.Next(); } Sort(folders); for (int ifold = 0; ifold < folders.GetCount(); ++ifold) { dgen.Generate(x, qtf, folders[ifold], lang, 2, String("/") + FormatInt(di) + "/[" + folders[ifold] + "]"); } d = AppendFileName(uppsrc, "plugin/*.*"); p = "plugin/"; } tt[di].text.Replace("[s5; <#packages#>]", qtf); String toc_qtf; const auto& headers = dgen.GetHeaders(); for (int i = 0; i < headers.GetCount(); i++) { toc_qtf << "[s0;l128; [^topic`:`/`/uppweb`/www`/documentation`$en`-us`#4`_" << IntStr(i + 1) << "^ " << headers[i] << "]&]"; } tt[di].text.Replace("[s0;l128; <#packages`_toc#>&]", toc_qtf); } } bi << BarLink(Www("Tutorials", lang), t_("Tutorials")); bi << BarLink(Www("UppHub", lang, "topic://ide/app/"), t_("UppHub")); bi << BarLink(Www("Roadmap", lang), t_("Status & Roadmap")); bi << BarLink(Www("FAQ", lang), t_("FAQ")); bi << BarLink(Www("About", lang, "topic://ide/app/"), t_("Authors & License")); bi << BarLink("https://www.ultimatepp.org/forums", t_("Forums")); // bcom << BarLink(Www("mailing"), "Mailing lists"); // bi << BarLink("http://www.ultimatepp.org/wiki/index.php", "Wiki"); bi << BarLink(Www("Funding", lang), t_("Funding U++")); // bcom << BarLink(Www("helpus"), "Getting involved"); // bcom << BarLink("mailto: upp@ntllib.org", "Contact developers"); bsearch << BarCaption(t_("Search on this site")); bsearch << SearchBar("ultimatepp.org"); #if 0 blang << BarCaption(t_("Language")); blang << Htmls("<div id=\"langbox\" style=\"display:none;\">") + BarItem(HtmlPackedTable().Width(-100) / HtmlRow() / ( HtmlCell() / Wimg(WWW::Language) + HtmlCell() / Htmls("<div id=\"langs\">"+GetNativeLangName(lang)+"</div>") ),"border: 0px solid black;" "padding-left:6px; padding-right:0px;" "padding-top:4px; padding-bottom:4px;" ) + Htmls("</div>"); #endif HtmlTag bf = HtmlPackedTable() .Width(-100) .BgColor(White) .Attr("style", "border-style: solid; border-width: 1px; border-color: #6E89AE;" "padding: 0px"); String div = HtmlTable().Border(0).Width(-100) / HtmlLine(); bar[i] = bf / bi + div + // bf / bex + div + // bf / bdoc + div + // bf / bcom + div + // bf / bcon + div + bf / bsearch + div; // bf / blang + div; } SetLanguage(currentLang); for(int i = 0; i < tt.GetCount(); i++) { String topic = tt.GetKey(i); links.Add(topic, topic == "topic://uppweb/www/index$en-us" ? "index.html" : memcmp(topic, "topic://", 8) ? topic : TopicFileNameHtml(topic)); } String svntableStr = DeQtf("[svntable]"); for(int i = 0; i < tt.GetCount(); i++) { if (tt[i].title == "Svn releases") tt[i].text.Replace(svntableStr, SvnChanges(svnlog, 300, "")); else if (tt[i].title == "Svn Web releases") tt[i].text.Replace(svntableStr, SvnChanges(svnlog, 100, "uppweb")); else if (tt[i].title == "Svn Bazaar releases") tt[i].text.Replace(svntableStr, SvnChanges(svnlog, 100, "bazaar")); else if (tt[i].title == "Svn Upp releases") tt[i].text.Replace(svntableStr, SvnChanges(svnlog, 100, "uppsrc")); else if (tt[i].title == "Svn major releases") tt[i].text.Replace(svntableStr, SvnChanges(svnlog, 300, "", true)); else if (tt[i].title == "Svn Web major releases") tt[i].text.Replace(svntableStr, SvnChanges(svnlog, 100, "uppweb", true)); else if (tt[i].title == "Svn Bazaar major releases") tt[i].text.Replace(svntableStr, SvnChanges(svnlog, 100, "bazaar", true)); else if (tt[i].title == "Svn Upp major releases") tt[i].text.Replace(svntableStr, SvnChanges(svnlog, 100, "uppsrc", true)); else if(tt[i].title == "Nightly builds" || tt[i].title.Find("Download") >= 0) tt[i].text.Replace(String("<#downloads#>"), downloads); else if (links[i].Find("index") >= 0) { String win32 = "upp-win32-RELEASE.exe"; String win32release = win32; win32release.Replace(String("RELEASE"), release); String x11 = "upp-x11-src-RELEASE.tar.gz"; String x11release = x11; x11release.Replace(String("RELEASE"), release); tt[i].text.Replace(DeQtf(x11), DeQtf(x11release)); tt[i].text.Replace(DeQtf(win32), DeQtf(win32release)); } } for(int i = 0; i < reflink.GetCount(); i++) { String l = reflink.GetKey(i); String lbl = Filter(l, CharFilterLbl); String f = links.Get(reflink[i], Null) + '#' + lbl; links.Add(l, f); static const char *x[] = { "::struct", "::class", "::union" }; for(int ii = 0; ii < 3; ii++) { String e = x[ii]; if(EndsWith(l, e)) { links.Add(l.Mid(0, l.GetLength() - e.GetLength()), f); } } labels.Add(l, lbl); } Date d = GetSysDate(); lastUpdate = HtmlItalic() / HtmlArial(8) / HtmlFontColor(Gray()) / (String().Cat() << "Last update " << GetSysDate()); // Vector<String> amazon = Split(LoadFile(GetRcFile("amazon.txt")), '\n');//440 Vector <SvnBazaarItems> bazaarItems = SvnBazaarList(bazaar, svnlog); // To fill Bazaar page with real release dates and size for(int i = 0; i < tt.GetCount(); i++) { if (tt[i].title == "Bazaar") { String page = tt[i]; for (int j = 0; j < bazaarItems.GetCount(); ++j) { int pos = page.Find("2 " + DeQtf(bazaarItems[j].name) + "]]"); if (pos < 0) pos = page.Find("2 " + DeQtf(bazaarItems[j].name) + "\r\n]]"); if (pos >= 0) { pos = page.Find("::", pos); pos = page.Find("::", pos); pos = page.Find(":: [s0;=2", pos); if (pos >= 0) pos += (int)strlen(":: [s0;=2"); String strDate; if (pos >= 0) { Time t = bazaarItems[j].lastChange; if (IsNull(t)) strDate = "Not in Svn log"; else strDate = Format("%Mon", t.month) + " " + FormatInt(t.year); page.Insert(pos, " " + strDate); } pos = page.Find(":: [s0;=2", pos); if (pos >= 0) pos += (int)strlen(":: [s0;=2"); String strSz; if (pos >= 0) { if (bazaarItems[j].size == 0) strSz = "Not in Svn log"; else strSz = BytesToString(bazaarItems[j].size); page.Insert(pos, " " + strSz); } } } tt[i].text = page; } } Vector <int> ttPdfId; for (int i = 0; i < ttFullIds.GetCount(); ++i) { Vector<int> ids; ids.Clear(); String sid, fullTitle; sid.Clear(); fullTitle.Clear(); int level = 0; for (const char *idStr = ttFullIds[i].Begin() + 1; *idStr; idStr++) { if (*idStr != '/') sid.Cat(*idStr); else { int id = ScanInt(sid); if (IsNull(id)) fullTitle << "/" << sid; else { if (id < 0 || id >= tt.GetCount()) { Cout() << "Problem in documentation"; return; } fullTitle << "/" << tt[id].title; level++; ids.Add(id); } sid.Clear(); } } int id = ScanInt(sid); if (IsNull(id)) fullTitle << "/" << sid; else { if (id < 0 || id >= tt.GetCount()) { Cout() << "Problem in documentation"; return; } fullTitle << "/" << tt[id].title; level++; } String com = ""; for (int iid = 0; iid < ids.GetCount(); ++iid) { if (ids[iid] == id) { com = "END LINK"; break; } if (tt[id].text == tt[ids[iid]].text) { com = "DUPLICATED"; break; } } if (fullTitle.Find("/Documentation") >= 0 && level <= 2 && com != "DUPLICATED") { Cout() << i << "\t(" << level << ")\t" << com << "\n"; Cout() << ttFullIds[i] << "\n"; Cout() << fullTitle << "\n"; ttPdfId << id; } } LLOG("G: " << MemoryUsedKb()); if (outPdf) { Cout() << "\nCreating pdf (" << tt.GetCount() << " items)\n"; PdfDraw pdf; //for(int i = 0; i < tt.GetCount(); i++) for(int i = 0; i < ttPdfId.GetCount(); i++) QtfAsPdf(pdf, tt[ttPdfId[i]]); SaveFile(AppendFileName(pdfdir, "Upp.pdf"), pdf.Finish()); } LLOG("H: " << MemoryUsedKb()); if (outHtml) { Cout() << "\nCreating htmls\n"; for(int i = 0; i < tt.GetCount(); i++) ExportPage(i); SetLanguage(currentLang); LLOG("I: " << MemoryUsedKb()); FileCopy(AppendFileName(uppbox, "uppweb/favicon.png"), AppendFileName(targetdir, "favicon.png")); // SaveFile(AppendFileName(targetdir, "favicon.ico"), LoadFile(AppendFileName(uppsrc, "ide/ide.ico"))); SaveFile(AppendFileName(targetdir, "stats.html"), HtmlLink("https://www.mygooglepagerank.com", "_blank") / "<img src=\"https://www.mygooglepagerank.com/PRimage.php?url=http://upp.sf.net\" " "border=\"0\" width=\"66\" height=\"13\" " "alt=\"Google PageRank™ - Post your PR with MyGooglePageRank.com\">" + "<noscript>" + HtmlLink("https://www.mygooglepagerank.com").Title("My Google Page Rank") / "My Google Page Rank" + "</noscript>" + HtmlLink("https://www.mygooglepagerank.com", "_blank") / "<img src=\"http://www.mygooglepagerank.com/PRimage.php?url=http://www.ultimatepp.org\" " "border=\"0\" width=\"66\" height=\"13\" " "alt=\"Google PageRank™ - Post your PR with MyGooglePageRank.com\">" + "<noscript>" + HtmlLink("https://www.mygooglepagerank.com").Title("My Google Page Rank") / "My Google Page Rank" + "</noscript>" ); } Cout() << "Finished OK\n"; }