X-Git-Url: https://git.decadent.org.uk/gitweb/?p=videolink.git;a=blobdiff_plain;f=webdvd.cpp;h=e3c02e90f271e757283ddf4f667d5ab8e5296193;hp=4d951ee9120f5502b0bec78b4aedcc92e2c16012;hb=cdd14ea76afc16f91e09323362468eaf5d0bcf9e;hpb=8d0ade32b10e040e462b244332afad1a0961d25b diff --git a/webdvd.cpp b/webdvd.cpp index 4d951ee..e3c02e9 100644 --- a/webdvd.cpp +++ b/webdvd.cpp @@ -2,6 +2,7 @@ // See the file "COPYING" for licence details. #include +#include #include #include #include @@ -9,12 +10,16 @@ #include #include #include +#include #include #include -#include + +#include #include +#include +#include #include #include @@ -24,6 +29,7 @@ #include #include #include +#include #include #include #include @@ -36,19 +42,23 @@ #include #include // required before nsILink.h #include +#include +#include #include #include +#include #include #include -#include "browserwidget.hpp" -#include "childiterator.hpp" +#include "browser_widget.hpp" +#include "child_iterator.hpp" #include "dvd.hpp" -#include "framebuffer.hpp" -#include "linkiterator.hpp" +#include "link_iterator.hpp" #include "pixbufs.hpp" -#include "stylesheets.hpp" +#include "style_sheets.hpp" +#include "temp_file.hpp" #include "video.hpp" +#include "x_frame_buffer.hpp" #include "xpcom_support.hpp" using xpcom_support::check; @@ -104,6 +114,7 @@ namespace { rectangle result; + // Start with this element's bounding box nsCOMPtr box; check(ns_doc->GetBoxObjectFor(elem, getter_AddRefs(box))); int width, height; @@ -114,7 +125,8 @@ namespace result.right = result.left + width; result.bottom = result.top + height; - for (ChildIterator it = ChildIterator(elem), end; it != end; ++it) + // Merge bounding boxes of all child elements + for (child_iterator it = child_iterator(elem), end; it != end; ++it) { nsCOMPtr child_node(*it); PRUint16 child_type; @@ -130,159 +142,248 @@ namespace return result; } - class WebDvdWindow : public Gtk::Window + struct dvd_contents + { + enum pgc_type { menu_pgc, title_pgc }; + typedef std::pair pgc_ref; + + struct menu + { + menu() + : vob_temp(new temp_file("webdvd-vob-")) + { + vob_temp->close(); + } + + boost::shared_ptr vob_temp; + std::vector entries; + }; + + struct title + { + explicit title(const std::string & vob_list) + : vob_list(vob_list) + {} + + std::string vob_list; + }; + + std::vector menus; + std::vector titles; + }; + + class webdvd_window : public Gtk::Window { public: - WebDvdWindow(int width, int height); - void add_page(const std::string & uri); + webdvd_window( + const video::frame_params & frame_params, + const std::string & main_page_uri, + const std::string & output_dir); private: - void add_video(const std::string & uri); + dvd_contents::pgc_ref add_menu(const std::string & uri); + dvd_contents::pgc_ref add_title(const std::string & uri); void load_next_page(); void on_net_state_change(const char * uri, gint flags, guint status); + bool process_page(); void save_screenshot(); void process_links(nsIPresShell * pres_shell, nsIPresContext * pres_context, nsIDOMWindow * dom_window); - void generate_dvdauthor_file(); + void generate_dvd(); - enum ResourceType { page_resource, video_resource }; - typedef std::pair<ResourceType, int> ResourceEntry; - int width_, height_; - BrowserWidget browser_widget_; + video::frame_params frame_params_; + std::string output_dir_; + browser_widget browser_widget_; nsCOMPtr<nsIStyleSheet> stylesheet_; + + dvd_contents contents_; + typedef std::map<std::string, dvd_contents::pgc_ref> resource_map_type; + resource_map_type resource_map_; + std::queue<std::string> page_queue_; - std::map<std::string, ResourceEntry> resource_map_; - std::vector<std::vector<std::string> > page_links_; - std::vector<std::string> video_paths_; - bool loading_; + bool pending_window_update_; int pending_req_count_; - struct link_state; - std::auto_ptr<link_state> link_state_; + bool have_tweaked_page_; + std::auto_ptr<temp_file> background_temp_; + struct page_state; + std::auto_ptr<page_state> page_state_; }; - WebDvdWindow::WebDvdWindow(int width, int height) - : width_(width), height_(height), + webdvd_window::webdvd_window( + const video::frame_params & frame_params, + const std::string & main_page_uri, + const std::string & output_dir) + : frame_params_(frame_params), + output_dir_(output_dir), stylesheet_(load_css("file://" WEBDVD_LIB_DIR "/webdvd.css")), - loading_(false), - pending_req_count_(0) + pending_window_update_(false), + pending_req_count_(0), + have_tweaked_page_(false) { - set_default_size(width, height); + set_size_request(frame_params_.width, frame_params_.height); + set_resizable(false); + add(browser_widget_); browser_widget_.show(); browser_widget_.signal_net_state().connect( - SigC::slot(*this, &WebDvdWindow::on_net_state_change)); + SigC::slot(*this, &webdvd_window::on_net_state_change)); + + add_menu(main_page_uri); + load_next_page(); } - void WebDvdWindow::add_page(const std::string & uri) + dvd_contents::pgc_ref webdvd_window::add_menu(const std::string & uri) { - if (resource_map_.insert( - std::make_pair(uri, ResourceEntry(page_resource, 0))) - .second) + dvd_contents::pgc_ref next_menu(dvd_contents::menu_pgc, + contents_.menus.size()); + std::pair<resource_map_type::iterator, bool> insert_result( + resource_map_.insert(std::make_pair(uri, next_menu))); + + if (!insert_result.second) + { + return insert_result.first->second; + } + else { page_queue_.push(uri); - if (!loading_) - load_next_page(); + contents_.menus.resize(contents_.menus.size() + 1); + return next_menu; } } - void WebDvdWindow::add_video(const std::string & uri) + dvd_contents::pgc_ref webdvd_window::add_title(const std::string & uri) { - if (resource_map_.insert( - std::make_pair(uri, ResourceEntry(video_resource, - video_paths_.size() + 1))) - .second) + dvd_contents::pgc_ref next_title(dvd_contents::title_pgc, + contents_.titles.size()); + std::pair<resource_map_type::iterator, bool> insert_result( + resource_map_.insert(std::make_pair(uri, next_title))); + + if (!insert_result.second) + { + return insert_result.first->second; + } + else { - // FIXME: Should accept some slightly different URI prefixes - // (e.g. file://localhost/) and decode any URI-escaped - // characters in the path. - assert(uri.compare(0, 8, "file:///") == 0); - video_paths_.push_back(uri.substr(7)); + Glib::ustring hostname; + std::string filename(Glib::filename_from_uri(uri, hostname)); + // FIXME: Should check the hostname + + std::string vob_list; + + // Store a reference to a linked VOB file, or the contents + // of a linked VOB list file. + if (filename.compare(filename.size() - 4, 4, ".vob") == 0) + { + if (!Glib::file_test(filename, Glib::FILE_TEST_IS_REGULAR)) + throw std::runtime_error( + filename + " is missing or not a regular file"); + vob_list + .append("<vob file='") + // FIXME: Should XML-escape the path + .append(filename) + .append("'/>\n"); + } + else + { + assert(filename.compare(filename.size() - 8, 8, ".voblist") + == 0); + // TODO: Validate the file contents + vob_list.assign(Glib::file_get_contents(filename)); + } + + contents_.titles.push_back(dvd_contents::title(vob_list)); + return next_title; } } - void WebDvdWindow::load_next_page() + void webdvd_window::load_next_page() { - loading_ = true; - assert(!page_queue_.empty()); const std::string & uri = page_queue_.front(); std::cout << "loading " << uri << std::endl; - std::size_t page_count = page_links_.size(); - resource_map_[uri].second = ++page_count; - page_links_.resize(page_count); browser_widget_.load_uri(uri); } - void WebDvdWindow::on_net_state_change(const char * uri, + void webdvd_window::on_net_state_change(const char * uri, gint flags, guint status) { - enum { - process_nothing, - process_new_page, - process_current_link - } action = process_nothing; +# ifdef DEBUG_ON_NET_STATE_CHANGE + std::cout << "webdvd_window::on_net_state_change("; + if (uri) + std::cout << '"' << uri << '"'; + else + std::cout << "NULL"; + std::cout << ", "; + { + gint flags_left = flags; + static const struct { + gint value; + const char * name; + } flag_names[] = { + { GTK_MOZ_EMBED_FLAG_START, "STATE_START" }, + { GTK_MOZ_EMBED_FLAG_REDIRECTING, "STATE_REDIRECTING" }, + { GTK_MOZ_EMBED_FLAG_TRANSFERRING, "STATE_TRANSFERRING" }, + { GTK_MOZ_EMBED_FLAG_NEGOTIATING, "STATE_NEGOTIATING" }, + { GTK_MOZ_EMBED_FLAG_STOP, "STATE_STOP" }, + { GTK_MOZ_EMBED_FLAG_IS_REQUEST, "STATE_IS_REQUEST" }, + { GTK_MOZ_EMBED_FLAG_IS_DOCUMENT, "STATE_IS_DOCUMENT" }, + { GTK_MOZ_EMBED_FLAG_IS_NETWORK, "STATE_IS_NETWORK" }, + { GTK_MOZ_EMBED_FLAG_IS_WINDOW, "STATE_IS_WINDOW" } + }; + for (int i = 0; i != sizeof(flag_names)/sizeof(flag_names[0]); ++i) + { + if (flags & flag_names[i].value) + { + std::cout << flag_names[i].name; + flags_left -= flag_names[i].value; + if (flags_left) + std::cout << " | "; + } + } + if (flags_left) + std::cout << "0x" << std::setbase(16) << flags_left; + } + std::cout << ", " << "0x" << std::setbase(16) << status << ")\n"; +# endif // DEBUG_ON_NET_STATE_CHANGE if (flags & GTK_MOZ_EMBED_FLAG_IS_REQUEST) { if (flags & GTK_MOZ_EMBED_FLAG_START) ++pending_req_count_; + if (flags & GTK_MOZ_EMBED_FLAG_STOP) { assert(pending_req_count_ != 0); --pending_req_count_; } - if (pending_req_count_ == 0 && link_state_.get()) - action = process_current_link; } - if (flags & GTK_MOZ_EMBED_FLAG_STOP - && flags & GTK_MOZ_EMBED_FLAG_IS_WINDOW) - action = process_new_page; + if (flags & GTK_MOZ_EMBED_FLAG_IS_DOCUMENT + && flags & GTK_MOZ_EMBED_FLAG_START) + { + pending_window_update_ = true; + have_tweaked_page_ = false; + } - if (action != process_nothing) + if (flags & GTK_MOZ_EMBED_FLAG_IS_WINDOW + && flags & GTK_MOZ_EMBED_FLAG_STOP) { - assert(loading_ && !page_queue_.empty()); - assert(pending_req_count_ == 0); + // Check whether the load was successful, ignoring this + // pseudo-error. + if (status != NS_IMAGELIB_ERROR_LOAD_ABORTED) + check(status); + + pending_window_update_ = false; + } + if (pending_req_count_ == 0 && !pending_window_update_) + { try { - // Check whether the load was successful, ignoring this - // pseudo-error. - if (status != NS_IMAGELIB_ERROR_LOAD_ABORTED) - check(status); - - nsCOMPtr<nsIWebBrowser> browser( - browser_widget_.get_browser()); - nsCOMPtr<nsIDocShell> doc_shell(do_GetInterface(browser)); - assert(doc_shell); - nsCOMPtr<nsIPresShell> pres_shell; - check(doc_shell->GetPresShell(getter_AddRefs(pres_shell))); - nsCOMPtr<nsIPresContext> pres_context; - check(doc_shell->GetPresContext( - getter_AddRefs(pres_context))); - nsCOMPtr<nsIDOMWindow> dom_window; - check(browser->GetContentDOMWindow( - getter_AddRefs(dom_window))); - - if (action == process_new_page) - { - apply_style_sheet(stylesheet_, pres_shell); - save_screenshot(); - } - process_links(pres_shell, pres_context, dom_window); - if (!link_state_.get()) - { - page_queue_.pop(); - if (page_queue_.empty()) - { - generate_dvdauthor_file(); - Gtk::Main::quit(); - } - else - load_next_page(); - } + if (!process_page()) + Gtk::Main::quit(); } catch (std::exception & e) { @@ -296,35 +397,119 @@ namespace } } - void WebDvdWindow::save_screenshot() + bool webdvd_window::process_page() + { + assert(!page_queue_.empty()); + + nsCOMPtr<nsIWebBrowser> browser(browser_widget_.get_browser()); + nsCOMPtr<nsIDocShell> doc_shell(do_GetInterface(browser)); + assert(doc_shell); + nsCOMPtr<nsIPresShell> pres_shell; + check(doc_shell->GetPresShell(getter_AddRefs(pres_shell))); + nsCOMPtr<nsIPresContext> pres_context; + check(doc_shell->GetPresContext(getter_AddRefs(pres_context))); + nsCOMPtr<nsIDOMWindow> dom_window; + check(browser->GetContentDOMWindow(getter_AddRefs(dom_window))); + + // If we haven't done so already, apply the stylesheet and + // disable scrollbars. + if (!have_tweaked_page_) + { + apply_style_sheet(stylesheet_, pres_shell); + + // This actually only needs to be done once. + nsCOMPtr<nsIDOMBarProp> dom_bar_prop; + check(dom_window->GetScrollbars(getter_AddRefs(dom_bar_prop))); + check(dom_bar_prop->SetVisible(false)); + + have_tweaked_page_ = true; + + // Might need to wait a while for things to load or more + // likely for a re-layout. + if (pending_req_count_ > 0) + return true; + } + + // All further work should only be done if we're not in preview mode. + if (!output_dir_.empty()) + { + // If we haven't already started work on this menu, save a + // screenshot of its normal appearance. + if (!page_state_.get()) + save_screenshot(); + + // Start or continue processing links. + process_links(pres_shell, pres_context, dom_window); + + // If we've finished work on the links, move on to the + // next page, if any, or else generate the DVD filesystem. + if (!page_state_.get()) + { + page_queue_.pop(); + if (page_queue_.empty()) + { + generate_dvd(); + return false; + } + else + { + load_next_page(); + } + } + } + + return true; + } + + void webdvd_window::save_screenshot() { - char filename[25]; - std::sprintf(filename, "page_%06d_back.png", page_links_.size()); Glib::RefPtr<Gdk::Window> window(get_window()); assert(window); window->process_updates(true); - std::cout << "saving " << filename << std::endl; + + background_temp_.reset(new temp_file("webdvd-back-")); + background_temp_->close(); + std::cout << "saving " << background_temp_->get_name() << std::endl; Gdk::Pixbuf::create(Glib::RefPtr<Gdk::Drawable>(window), window->get_colormap(), - 0, 0, 0, 0, width_, height_) - ->save(filename, "png"); + 0, 0, 0, 0, + frame_params_.width, frame_params_.height) + ->save(background_temp_->get_name(), "png"); } - struct WebDvdWindow::link_state + struct webdvd_window::page_state { + page_state(nsIDOMDocument * doc, int width, int height) + : diff_pixbuf(Gdk::Pixbuf::create( + Gdk::COLORSPACE_RGB, + true, 8, // has_alpha, bits_per_sample + width, height)), + spumux_temp("webdvd-spumux-"), + links_temp("webdvd-links-"), + link_num(0), + links_it(doc), + link_changing(false) + { + spumux_temp.close(); + links_temp.close(); + } + Glib::RefPtr<Gdk::Pixbuf> diff_pixbuf; + temp_file spumux_temp; std::ofstream spumux_file; + temp_file links_temp; + int link_num; - LinkIterator links_it, links_end; + link_iterator links_it, links_end; rectangle link_rect; bool link_changing; Glib::RefPtr<Gdk::Pixbuf> norm_pixbuf; }; - void WebDvdWindow::process_links(nsIPresShell * pres_shell, + void webdvd_window::process_links(nsIPresShell * pres_shell, nsIPresContext * pres_context, nsIDOMWindow * dom_window) { @@ -347,37 +532,27 @@ namespace check(doc_view->GetDefaultView(getter_AddRefs(view))); // Set up or recover our iteration state. - std::auto_ptr<link_state> state(link_state_); + std::auto_ptr<page_state> state(page_state_); if (!state.get()) { - state.reset(new link_state); - - state->diff_pixbuf = Gdk::Pixbuf::create(Gdk::COLORSPACE_RGB, - true, // has_alpha - 8, // bits_per_sample - width_, height_); - - char spumux_filename[20]; - std::sprintf(spumux_filename, - "page_%06d.spumux", page_links_.size()); - state->spumux_file.open(spumux_filename); + state.reset( + new page_state( + basic_doc, frame_params_.width, frame_params_.height)); + + state->spumux_file.open(state->spumux_temp.get_name().c_str()); state->spumux_file << "<subpictures>\n" " <stream>\n" " <spu force='yes' start='00:00:00.00'\n" - " highlight='page_" << std::setfill('0') - << std::setw(6) << page_links_.size() - << "_links.png'\n" - " select='page_" << std::setfill('0') - << std::setw(6) << page_links_.size() - << "_links.png'>\n"; - - state->link_num = 0; - state->links_it = LinkIterator(basic_doc); - state->link_changing = false; + " highlight='" << state->links_temp.get_name() << "'\n" + " select='" << state->links_temp.get_name() << "'>\n"; } - - rectangle window_rect = { 0, 0, width_, height_ }; + + rectangle window_rect = { + 0, 0, frame_params_.width, frame_params_.height + }; + + int menu_num = resource_map_[page_queue_.front()].second; for (/* no initialisation */; state->links_it != state->links_end; @@ -422,19 +597,30 @@ namespace { if (state->link_num == dvd::menu_buttons_max) std::cerr << "No more than " << dvd::menu_buttons_max - << " buttons can be placed on a page\n"; + << " buttons can be placed on a menu\n"; std::cerr << "Ignoring link to " << uri_string << "\n"; continue; } + state->spumux_file << + " <button x0='" << state->link_rect.left << "'" + " y0='" << state->link_rect.top << "'" + " x1='" << state->link_rect.right - 1 << "'" + " y1='" << state->link_rect.bottom - 1 << "'/>\n"; + // Check whether this is a link to a video or a page then - // add it to the known resources if not already seen. + // add it to the known resources if not already seen; then + // add it to the menu entries. nsCString path; check(uri->GetPath(path)); + dvd_contents::pgc_ref dest_pgc; // FIXME: This is a bit of a hack. Perhaps we could decide // later based on the MIME type determined by Mozilla? - if (path.Length() > 4 - && std::strcmp(path.EndReading() - 4, ".vob") == 0) + if ((path.Length() > 4 + && std::strcmp(path.EndReading() - 4, ".vob") == 0) + || (path.Length() > 8 + && std::strcmp(path.EndReading() - 8, ".voblist") + == 0)) { PRBool is_file; check(uri->SchemeIs("file", &is_file)); @@ -444,12 +630,13 @@ namespace << " scheme\n"; continue; } - add_video(uri_sans_fragment); + dest_pgc = add_title(uri_sans_fragment); } else { - add_page(uri_sans_fragment); + dest_pgc = add_menu(uri_sans_fragment); } + contents_.menus[menu_num].entries.push_back(dest_pgc); nsCOMPtr<nsIContent> content(do_QueryInterface(node)); assert(content); @@ -501,7 +688,7 @@ namespace if (pending_req_count_ > 0) { state->link_changing = true; - link_state_ = state; + page_state_ = state; return; } } @@ -526,70 +713,100 @@ namespace state->link_rect.top, state->link_rect.right - state->link_rect.left, state->link_rect.bottom - state->link_rect.top); - - state->spumux_file << - " <button x0='" << state->link_rect.left << "'" - " y0='" << state->link_rect.top << "'" - " x1='" << state->link_rect.right - 1 << "'" - " y1='" << state->link_rect.bottom - 1 << "'/>\n"; - - // Add to the page's links, ignoring any fragment (for now). - page_links_.back().push_back(uri_sans_fragment); } quantise_rgba_pixbuf(state->diff_pixbuf, dvd::button_n_colours); - char filename[25]; - std::sprintf(filename, "page_%06d_links.png", page_links_.size()); - std::cout << "saving " << filename << std::endl; - state->diff_pixbuf->save(filename, "png"); + std::cout << "saving " << state->links_temp.get_name() + << std::endl; + state->diff_pixbuf->save(state->links_temp.get_name(), "png"); state->spumux_file << " </spu>\n" " </stream>\n" "</subpictures>\n"; + + state->spumux_file.close(); + + // TODO: if (!state->spumux_file) throw ... + + { + std::ostringstream command_stream; + command_stream << "pngtopnm " + << background_temp_->get_name() + << " | ppmtoy4m -v0 -n1 -F" + << frame_params_.rate_numer + << ":" << frame_params_.rate_denom + << " -A" << frame_params_.pixel_ratio_width + << ":" << frame_params_.pixel_ratio_height + << (" -Ip -S420_mpeg2" + " | mpeg2enc -v0 -f8 -a2 -o/dev/stdout" + " | mplex -v0 -f8 -o/dev/stdout /dev/stdin" + " | spumux -v0 -mdvd ") + << state->spumux_temp.get_name() + << " > " + << contents_.menus[menu_num].vob_temp->get_name(); + std::string command(command_stream.str()); + const char * argv[] = { + "/bin/sh", "-c", command.c_str(), 0 + }; + std::cout << "running " << argv[2] << std::endl; + int command_result; + Glib::spawn_sync(".", + Glib::ArrayHandle<std::string>( + argv, sizeof(argv)/sizeof(argv[0]), + Glib::OWNERSHIP_NONE), + Glib::SPAWN_STDOUT_TO_DEV_NULL, + SigC::Slot0<void>(), + 0, 0, + &command_result); + if (command_result != 0) + throw std::runtime_error("spumux pipeline failed"); + } } - void generate_page_dispatch(std::ostream &, int indent, - int first_page, int last_page); + void generate_menu_dispatch(std::ostream &, int indent, + int first_menu, int last_menu); - void WebDvdWindow::generate_dvdauthor_file() + void webdvd_window::generate_dvd() { - std::ofstream file("webdvd.dvdauthor"); + temp_file temp("webdvd-dvdauthor-"); + temp.close(); + std::ofstream file(temp.get_name().c_str()); // We generate code that uses registers in the following way: // - // g0: link destination (when jumping to menu 1), then scratch + // g0: button destination (when jumping to menu 1), then scratch // g1: current location // g2-g11: location history (g2 = most recent) // g12: location that last linked to a video // // All locations are divided into two bitfields: the least // significant 10 bits are a page/menu number and the most - // significant 6 bits are a link/button number. This is - // chosen for compatibility with the encoding of the s8 - // (button) register. + // significant 6 bits are a link/button number, and numbering + // starts at 1, not 0. This is done for compatibility with + // the encoding of the s8 (button) register. // - static const int link_mult = dvd::reg_s8_button_mult; - static const int page_mask = link_mult - 1; - static const int link_mask = (1 << dvd::reg_bits) - link_mult; + static const int button_mult = dvd::reg_s8_button_mult; + static const int menu_mask = button_mult - 1; + static const int button_mask = (1 << dvd::reg_bits) - button_mult; + static const int location_bias = button_mult + 1; file << "<dvdauthor>\n" " <vmgm>\n" " <menus>\n"; - for (std::size_t page_num = 1; - page_num <= page_links_.size(); - ++page_num) + for (std::size_t menu_num = 0; + menu_num != contents_.menus.size(); + ++menu_num) { - std::vector<std::string> & page_links = - page_links_[page_num - 1]; + dvd_contents::menu & menu = contents_.menus[menu_num]; - if (page_num == 1) + if (menu_num == 0) { - // This is the first page (root menu) which needs to - // include initialisation and dispatch code. + // This is the first (title) menu, which needs to include + // initialisation and dispatch code. file << " <pgc entry='title'>\n" @@ -597,9 +814,9 @@ namespace // Has the location been set yet? " if (g1 eq 0)\n" " {\n" - // Initialise the current location to first link on - // this page. - " g1 = " << 1 * link_mult + 1 << ";\n" + // Initialise the current location to first button on + // this menu. + " g1 = " << location_bias << ";\n" " }\n" " else\n" " {\n" @@ -608,7 +825,7 @@ namespace " {\n" // First update the history. // Does link go to the last page in the history? - " if (((g0 ^ g2) & " << page_mask + " if (((g0 ^ g2) & " << menu_mask << ") == 0)\n" // It does; we treat this as going back and pop the old // location off the history stack into the current @@ -629,15 +846,16 @@ namespace " }\n" " }\n" // Find the target page number. - " g0 = g1 & " << page_mask << ";\n"; + " g0 = g1 & " << menu_mask << ";\n"; // There seems to be no way to perform a computed jump, // so we generate all possible jumps and a binary search // to select the correct one. - generate_page_dispatch(file, 12, 1, page_links_.size()); + generate_menu_dispatch(file, 12, + 0, contents_.menus.size() - 1); file << " }\n"; } - else // page_num != 1 + else // menu_num != 0 { file << " <pgc>\n" @@ -647,30 +865,35 @@ namespace file << // Clear link indicator and highlight the // appropriate link/button. - " g0 = 0; s8 = g1 & " << link_mask << ";\n" + " g0 = 0; s8 = g1 & " << button_mask << ";\n" " </pre>\n" - " <vob file='page_" - << std::setfill('0') << std::setw(6) << page_num - << ".vob'/>\n"; + " <vob file='" + << menu.vob_temp->get_name() << "'/>\n"; - for (std::size_t link_num = 1; - link_num <= page_links.size(); - ++link_num) + for (std::size_t button_num = 0; + button_num != menu.entries.size(); + ++button_num) { - file << - " <button>" + file << " <button> " // Update current location. - " g1 = " << link_num * link_mult + page_num << ";"; + " g1 = " + << location_bias + button_num * button_mult + menu_num + << ";"; // Jump to appropriate resource. - const ResourceEntry & resource_loc = - resource_map_[page_links[link_num - 1]]; - if (resource_loc.first == page_resource) - file << - " g0 = " << 1 * link_mult + resource_loc.second << ";" - " jump menu 1;"; - else if (resource_loc.first == video_resource) - file << " jump title " << resource_loc.second << ";"; + if (menu.entries[button_num].first == dvd_contents::menu_pgc) + { + file << " g0 = " + << location_bias + menu.entries[button_num].second + << "; jump menu 1;"; + } + else + { + assert(menu.entries[button_num].first + == dvd_contents::title_pgc); + file << " jump title " + << 1 + menu.entries[button_num].second << ";"; + } file << " </button>\n"; } @@ -682,11 +905,12 @@ namespace " </menus>\n" " </vmgm>\n"; - // Generate a titleset for each video. This appears to make - // jumping to titles a whole lot simpler. - for (std::size_t video_num = 1; - video_num <= video_paths_.size(); - ++video_num) + // Generate a titleset for each title. This appears to make + // jumping to titles a whole lot simpler (but limits us to 99 + // titles). + for (std::size_t title_num = 0; + title_num != contents_.titles.size(); + ++title_num) { file << " <titleset>\n" @@ -701,14 +925,12 @@ namespace " <pgc>\n" // Record calling page/menu. " <pre> g12 = g1; </pre>\n" - // FIXME: Should XML-escape the path - " <vob file='" << video_paths_[video_num - 1] - << "'/>\n" + << contents_.titles[title_num].vob_list << // If page/menu location has not been changed during the // video, change the location to be the following // link/button when returning to it. In any case, // return to a page/menu. - " <post> if (g1 eq g12) g1 = g1 + " << link_mult + " <post> if (g1 eq g12) g1 = g1 + " << button_mult << "; call menu; </post>\n" " </pgc>\n" " </titles>\n" @@ -717,76 +939,251 @@ namespace file << "</dvdauthor>\n"; - } - void generate_page_dispatch(std::ostream & file, int indent, - int first_page, int last_page) - { - if (first_page == 1 && last_page == 1) - { - // The dispatch code is *on* page 1 so we must not dispatch to - // page 1 since that would cause an infinite loop. This case - // should be unreachable if there is more than one page due - // to the following case. - } - else if (first_page == 1 && last_page == 2) + file.close(); + { - // dvdauthor doesn't allow empty blocks or null statements so - // when selecting between pages 1 and 2 we don't use an "else" - // part. We must use braces so that a following "else" will - // match the right "if". - file << std::setw(indent) << "" << "{\n" - << std::setw(indent) << "" << "if (g0 eq 2)\n" - << std::setw(indent + 2) << "" << "jump menu 2;\n" - << std::setw(indent) << "" << "}\n"; + const char * argv[] = { + "dvdauthor", + "-o", output_dir_.c_str(), + "-x", temp.get_name().c_str(), + 0 + }; + int command_result; + Glib::spawn_sync(".", + Glib::ArrayHandle<std::string>( + argv, sizeof(argv)/sizeof(argv[0]), + Glib::OWNERSHIP_NONE), + Glib::SPAWN_SEARCH_PATH + | Glib::SPAWN_STDOUT_TO_DEV_NULL, + SigC::Slot0<void>(), + 0, 0, + &command_result); + if (command_result != 0) + throw std::runtime_error("dvdauthor failed"); } - else if (first_page == last_page) + } + + void generate_menu_dispatch(std::ostream & file, int indent, + int first_menu, int last_menu) + { + if (first_menu == last_menu) { - file << std::setw(indent) << "" - << "jump menu " << first_page << ";\n"; + if (first_menu == 0) + { + // This dispatch code is generated *on* the first menu + // so don't create an infinite loop. + } + else + { + file << std::setw(indent) << "" + << "jump menu " << 1 + first_menu << ";\n"; + } } - else + else // first_menu != last_menu { - int middle = (first_page + last_page) / 2; - file << std::setw(indent) << "" << "if (g0 le " << middle << ")\n"; - generate_page_dispatch(file, indent + 2, first_page, middle); - file << std::setw(indent) << "" << "else\n"; - generate_page_dispatch(file, indent + 2, middle + 1, last_page); + if (first_menu == 0 && last_menu == 1) + { + // dvdauthor doesn't allow empty blocks or null + // statements so when selecting between the first 2 + // menus we don't use an "else" part. We must use + // braces so that a following "else" will match the + // right "if". + file << std::setw(indent) << "" << "{\n" + << std::setw(indent) << "" << "if (g0 eq 2)\n" + << std::setw(indent + 2) << "" << "jump menu 2;\n" + << std::setw(indent) << "" << "}\n"; + } + else + { + int middle = (first_menu + last_menu) / 2; + file << std::setw(indent) << "" << "if (g0 le " << 1 + middle + << ")\n"; + generate_menu_dispatch(file, indent + 2, + first_menu, middle); + file << std::setw(indent) << "" << "else\n"; + generate_menu_dispatch(file, indent + 2, + middle + 1, last_menu); + } } } + const video::frame_params & lookup_frame_params(const char * str) + { + assert(str); + static const struct { const char * str; bool is_ntsc; } + known_strings[] = { + { "NTSC", true }, + { "ntsc", true }, + { "PAL", false }, + { "pal", false }, + // For DVD purposes, SECAM can be treated identically to PAL. + { "SECAM", false }, + { "secam", false } + }; + for (std::size_t i = 0; + i != sizeof(known_strings)/sizeof(known_strings[0]); + ++i) + if (std::strcmp(str, known_strings[i].str) == 0) + return known_strings[i].is_ntsc ? + video::ntsc_params : video::pal_params; + throw std::runtime_error( + std::string("Invalid video standard: ").append(str)); + } + + void print_usage(std::ostream & stream, const char * command_name) + { + stream << "Usage: " << command_name + << (" [gtk-options] [--video-std std-name]" + " [--preview] menu-url [output-dir]\n"); + } + + void set_browser_preferences() + { + // Disable IE-compatibility kluge that causes backgrounds to + // sometimes/usually be missing from snapshots. This is only + // effective from Mozilla 1.8 onward. +# if MOZ_VERSION_MAJOR > 1 \ + || (MOZ_VERSION_MAJOR == 1 && MOZ_VERSION_MINOR >= 8) + nsCOMPtr<nsIPrefService> pref_service; + static const nsCID pref_service_cid = NS_PREFSERVICE_CID; + check(CallGetService<nsIPrefService>(pref_service_cid, + getter_AddRefs(pref_service))); + nsCOMPtr<nsIPrefBranch> pref_branch; + check(pref_service->GetDefaultBranch("layout", + getter_AddRefs(pref_branch))); + check(pref_branch->SetBoolPref( + "fire_onload_after_image_background_loads", + true)); +# endif + + // TODO: Set display resolution? Unfortunately Mozilla doesn't + // support non-square pixels (and neither do fontconfig or Xft + // anyway). + } + } // namespace int main(int argc, char ** argv) { - // Get dimensions - int width = video::pal_oscan_width, height = video::pal_oscan_height; - for (int i = 1; i < argc - 1; ++i) - if (std::strcmp(argv[i], "-geometry") == 0) + try + { + video::frame_params frame_params = video::pal_params; + bool preview_mode = false; + std::string menu_url; + std::string output_dir; + + // Do initial option parsing. We have to do this before + // letting Gtk parse the arguments since we may need to spawn + // Xvfb first. + int argi = 1; + while (argi != argc) { - std::sscanf(argv[i + 1], "%dx%d", &width, &height); - break; + if (std::strcmp(argv[argi], "--") == 0) + { + break; + } + else if (std::strcmp(argv[argi], "--help") == 0) + { + print_usage(std::cout, argv[0]); + return EXIT_SUCCESS; + } + else if (std::strcmp(argv[argi], "--preview") == 0) + { + preview_mode = true; + argi += 1; + } + else if (std::strcmp(argv[argi], "--video-std") == 0) + { + if (argi + 1 == argc) + { + std::cerr << "Missing argument to --video-std\n"; + print_usage(std::cerr, argv[0]); + return EXIT_FAILURE; + } + frame_params = lookup_frame_params(argv[argi + 1]); + argi += 2; + } + else + { + argi += 1; + } } - // A depth of 24 results in 8 bits each for RGB components, which - // translates into "enough" bits for YUV components. - const int depth = 24; - try - { - // Spawn Xvfb and set env variables so that Xlib will use it - FrameBuffer fb(width, height, depth); - setenv("XAUTHORITY", fb.get_x_authority().c_str(), true); - setenv("DISPLAY", fb.get_x_display().c_str(), true); + std::auto_ptr<x_frame_buffer> fb; + if (!preview_mode) + { + // Spawn Xvfb and set env variables so that Xlib will use it + // Use 8 bits each for RGB components, which should translate into + // "enough" bits for YUV components. + fb.reset(new x_frame_buffer(frame_params.width, + frame_params.height, + 3 * 8)); + setenv("XAUTHORITY", fb->get_authority().c_str(), true); + setenv("DISPLAY", fb->get_display().c_str(), true); + } - // Initialise Gtk and Mozilla + // Initialise Gtk Gtk::Main kit(argc, argv); - BrowserWidget::init(); - WebDvdWindow window(width, height); - for (int argi = 1; argi < argc; ++argi) - window.add_page(argv[argi]); - if (argc < 2) - window.add_page("about:"); + // Complete option parsing with Gtk's options out of the way. + argi = 1; + while (argi != argc) + { + if (std::strcmp(argv[argi], "--") == 0) + { + argi += 1; + break; + } + else if (std::strcmp(argv[argi], "--preview") == 0) + { + argi += 1; + } + else if (std::strcmp(argv[argi], "--video-std") == 0) + { + argi += 2; + } + else if (argv[argi][0] == '-') + { + std::cerr << "Invalid option: " << argv[argi] << "\n"; + print_usage(std::cerr, argv[0]); + return EXIT_FAILURE; + } + else + { + break; + } + } + + // Look for a starting URL or filename and (except in preview + // mode) an output directory after the options. + if (argc - argi != (preview_mode ? 1 : 2)) + { + print_usage(std::cerr, argv[0]); + return EXIT_FAILURE; + } + if (std::strstr(argv[argi], "://")) + { + // It appears to be an absolute URL, so use it as-is. + menu_url = argv[argi]; + } + else + { + // Assume it's a filename. Resolve it to an absolute URL. + std::string path(argv[argi]); + if (!Glib::path_is_absolute(path)) + path = Glib::build_filename(Glib::get_current_dir(), path); + menu_url = Glib::filename_to_uri(path); + } + if (!preview_mode) + output_dir = argv[argi + 1]; + + // Initialise Mozilla + browser_widget::initialiser browser_init; + set_browser_preferences(); + + // Run the browser/converter + webdvd_window window(frame_params, menu_url, output_dir); Gtk::Main::run(window); } catch (std::exception & e)