X-Git-Url: https://harishankar.org/repos/?p=biaweb2.git;a=blobdiff_plain;f=biawebdocumenttree.hpp;h=a9f1ccd90dd40cc6e6b7919f30c024783c06cecc;hp=608b70242179dc0056c8ada7318983891bd40812;hb=HEAD;hpb=1d1f04629b04c7e7efd14d8f281e17f5e491c888 diff --git a/biawebdocumenttree.hpp b/biawebdocumenttree.hpp index 608b702..a9f1ccd 100644 --- a/biawebdocumenttree.hpp +++ b/biawebdocumenttree.hpp @@ -2,14 +2,19 @@ #define __BIAWEBDOCUMENTTREE__ #include #include +#include #include +#include #include #include #include "biawebdocument.hpp" #include "biawebstrings.hpp" #include "biawebutil.hpp" +#include "biawebdoclist.hpp" +#include "biawebtemplate.hpp" +#include "biawebrss.hpp" -// to implement a document tree - both with or without subtrees +// class to implement a document tree - both with or without subtrees namespace biaweb { class DocumentTree { protected: @@ -38,7 +43,7 @@ namespace biaweb { // create new top level document tree DocumentTree (std::string title, std::string stub = "") { - this->title = title; + this->title = escape_html (title); // if stub is not empty set it if (stub != "") this->stub = stub; @@ -69,11 +74,11 @@ namespace biaweb { } // create the document index for this tree - void create_tree_html (std::string templatedir, std::string destdir); + void create_tree_html (Template *t, std::string destdir); // set the title void set_title (std::string title) { - this->title = title; + this->title = escape_html (title); // if no stub is set if (this->stub == "") this->stub = convert_title (title); @@ -161,7 +166,9 @@ namespace biaweb { // create the tree - the index file for this tree and all the documents and // the child trees recursively - using the template specified - void DocumentTree::create_tree_html (std::string templatedir, std::string destdir) { + void DocumentTree::create_tree_html (Template *tpl, std::string destdir) { + + // create a document to represent the index of the tree. std::unique_ptr index (new Document (this->title)); index.get()->set_index (); // set the file name path @@ -179,60 +186,110 @@ namespace biaweb { // Create a link to the index page and // If this tree has a parent, create a sidebar link to the level up std::unique_ptr bar1 (new SideBar()); - SideBarItem item0; - item0.set_sidebar_text (INDEX); - item0.set_sidebar_url (urlpath + "index.html"); + GenericLinkItem item0; + bar1.get()->set_title (tpl->get_stringbit (NAVIGATION)); + item0.set_item_text (tpl->get_stringbit (INDEX)); + item0.set_item_url (urlpath + "index.html"); bar1.get()->add_sidebar_item (item0); if (this->get_parent() != nullptr) { - SideBarItem item1; - item1.set_sidebar_text (GO_UP); - item1.set_sidebar_url (this->stub_hierarchy() + "index.html"); + GenericLinkItem item1; + item1.set_item_text (tpl->get_stringbit(GO_UP)); + item1.set_item_url (this->stub_hierarchy() + "index.html"); bar1.get()->add_sidebar_item (item1); } - index.get()->add_side_bar (*bar1.get()); - + // create a sidebar for the child levels if there are children std::unique_ptr bar2 (new SideBar ()); - bar2.get()->set_title (SUB_CAT + this->title); + bar2.get()->set_title (tpl->get_stringbit (SUB_CAT) + this->title); for (DocumentTree tree : this->children) { // we use site relative URLs that rely on the base href tag // so for biaweb generated sites, the base href tag should be // used in the main template - SideBarItem item (tree.get_title(), urlpath + + GenericLinkItem item (tree.get_title(), urlpath + tree.stub + "/" + "index.html"); bar2.get()->add_sidebar_item (item); } - index.get()->add_side_bar (*bar2.get()); // create the path and then the index file std::filesystem::create_directories (filepath); // Create the list of documents in this tree with links - // Reuse the sidebar class and sidebaritem class which is - // basically a list of links but instead of adding a sidebar - // add it to the content portion - SideBar article_list; - article_list.set_title (this->title + ": " + ART_LIST); + std::unique_ptr article_list (new DocList ()); + article_list.get()->set_title (this->title + ": " + tpl->get_stringbit (ARTICLES_LIST)); // sort the documents as per creation time and then add the document // links - newest documents should appear above older ones. sort_documents_creation_time (); + + // create the navigation bit + std::shared_ptr navbit (new NavigationBit ()); + auto par1 = this; + // get the link to each level in the hierarchy and add it as + // an inline list + while (par1 != nullptr) { + if (par1->parent != nullptr) + navbit.get()->add_link_item (GenericLinkItem(par1->title, + par1->stub_hierarchy() + par1->stub + "/index.html")); + else + navbit.get()->add_link_item (GenericLinkItem(tpl->get_stringbit(HOME), "index.html")); + par1 = par1->parent; + } + + // rss feed + std::unique_ptr feed (new RSSFeed ()); + feed.get()->set_pub_date (index.get()->get_creation_date()); + feed.get()->set_title (this->get_title()); + + // iterate through the documents and generate the document for (Document doc : this->docs) { - SideBarItem item; - item.set_sidebar_text (doc.get_title()); - item.set_sidebar_url (urlpath + doc.get_filename() + ".html"); - article_list.add_sidebar_item (item); - // output the document also, add the side bars + // Add the document to RSS feed + RSSFeedItem fitem (doc.get_title(), doc.get_meta_desc (), + urlpath + doc.get_filename() + ".html", + doc.get_creation_date()); + // If the items don't exceed max size of RSS feed + if (feed.get()->get_num_items() < MAX_RSS_FEED) + feed.get()->add_rss_item (fitem); + // Add the document details to the document list + DocListItem item (&doc, urlpath); + article_list.get()->add_document_item (item); + // output the document also, add the navigation bit and side bars + + doc.set_navigation_bit (*navbit.get()); doc.add_side_bar (*bar1.get()); doc.add_side_bar (*bar2.get()); - doc.output_to_html (templatedir, filepath); + doc.output_to_html (tpl, filepath); + } + + // output the rss feed + feed.get()->output_to_html (tpl, filepath); + + // add the side bars + index.get()->add_side_bar (*bar1.get()); + index.get()->add_side_bar (*bar2.get()); + + // sidebar for RSS feed + // if there are are items in the feed add the link + if (feed.get()->get_num_items () > 0) { + std::unique_ptr bar3 (new SideBar ()); + bar3.get()->set_title (tpl->get_stringbit (SUBSCRIBE)); + bar3.get()->add_sidebar_item ( + GenericLinkItem (tpl->get_stringbit(RSS_FEED), urlpath + "feed.xml")); + index.get()->add_side_bar (*bar3.get()); } - index.get()->set_content (this->summary + article_list.to_html(templatedir)); - index.get()->output_to_html (templatedir, filepath); + // add the navigation bit + index.get()->set_navigation_bit (*navbit.get()); + // index should contain the summary followed by the the article list + // and the sub categories + index.get()->set_content (this->summary + + article_list.get()->to_html(tpl) + + bar2.get()->to_html (tpl)); + + // output the index file + index.get()->output_to_html (tpl, filepath); - // create index for children + // recursively create index for children for (DocumentTree tree : this->children) - tree.create_tree_html (templatedir, destdir); + tree.create_tree_html (tpl, destdir); } // build a document tree from a filesystem path recursively @@ -252,32 +309,28 @@ namespace biaweb { this->add_child (doctree.get()); } - // add the regular files as documents in the tree - else if (fsitem.is_regular_file ()) { - // read the contents of the file - std::ifstream infile (fsitem.path().string()); - std::string infilestr ((std::istreambuf_iterator (infile)), - (std::istreambuf_iterator ())); - // if it is an index file (specially named index) add + // add the regular files as documents in the tree and not symlink + else if (fsitem.is_regular_file () && !fsitem.is_symlink()) { + // if it is an index file (specially named as index + // or index.md or whatever) directly add // the contents to the summary of the Doctree if (fsitem.path().stem().string() == "index") { + std::string infilestr = load_from_file (fsitem.path ()); this->set_markdown_summary (infilestr); } // else it is a non-index file- // create a Document and add it to the tree else { + std::ifstream infile (fsitem.path ()); std::shared_ptr doc - (new Document (fsitem.path().stem().string())); + (new Document (infile)); + infile.close (); - // file creation/modified date from system + // file modified date from system struct stat buf; if (stat (fsitem.path().string().c_str(), &buf) == 0) - { - doc.get()->set_creation_date (buf.st_ctim.tv_sec); doc.get()->set_modified_date (buf.st_mtim.tv_sec); - } - doc.get()->set_markdown_content (infilestr); this->add_document (doc.get()); } @@ -285,7 +338,7 @@ namespace biaweb { } } catch (std::filesystem::filesystem_error) { - std::cout << "No such path! Specify an existing file path" << std::endl; + std::cout << NO_SUCH_PATH_ERROR << std::endl; } // add the trees for the children recursively