/* Copyright (C) 2014 Valeriy Kamyshniy 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 2 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, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include #include "waves_ui.h" #include "pbd/file_utils.h" #include "ardour/filesystem_paths.h" #include "utils.h" #include "pbd/convert.h" #include "dbg_msg.h" using namespace PBD; using namespace ARDOUR; Gtk::Widget* WavesUI::create_widget (const XMLNode& definition, const XMLNodeMap& styles, std::map& named_widgets) { Gtk::Widget* child = NULL; std::string widget_type = definition.name(); std::string widget_id = xml_property (definition, "id", styles, ""); std::string text = xml_property (definition, "text", styles, ""); boost::replace_all(text, "\\n", "\n"); std::transform(widget_type.begin(), widget_type.end(), widget_type.begin(), ::toupper); if (widget_type == "BUTTON") { WavesButton& button = *manage (new WavesButton(text)); child = &button; button.set_border_width (xml_property (definition, "borderwidth", styles, "0").c_str()); button.set_border_color (xml_property (definition, "bordercolor", styles, "#000000").c_str()); } else if (widget_type == "ICONBUTTON") { WavesIconButton& iconbutton = *manage (new WavesIconButton); child = &iconbutton; iconbutton.set_border_width (xml_property (definition, "borderwidth", styles, "0").c_str()); iconbutton.set_border_color (xml_property (definition, "bordercolor", styles, "#000000").c_str()); std::string property = xml_property (definition, "normalicon", styles, ""); if (!property.empty ()) { iconbutton.set_normal_image(get_icon(property.c_str())); } property = xml_property (definition, "activeicon", styles, ""); if (!property.empty ()) { iconbutton.set_active_image(get_icon(property.c_str())); } property = xml_property (definition, "prelighticon", styles, ""); if (!property.empty ()) { iconbutton.set_prelight_image(get_icon(property.c_str())); } property = xml_property (definition, "inactiveicon", styles, ""); if (!property.empty ()) { iconbutton.set_inactive_image(get_icon(property.c_str())); } } else if (widget_type == "COMBOBOXTEXT") { child = manage (new Gtk::ComboBoxText); } else if (widget_type == "LABEL") { child = manage (new Gtk::Label(text)); } else if (widget_type == "LAYOUT") { child = manage (new Gtk::Layout); } else if (widget_type == "CANVAS") { std::map named_items; child = manage (new ArdourCanvas::GtkCanvas(definition, styles, named_items)); } else if (widget_type == "SCROLLEDWINDOW") { child = manage (new Gtk::ScrolledWindow); Gtk::PolicyType hscrollbar_policy = Gtk::POLICY_AUTOMATIC; Gtk::PolicyType vscrollbar_policy = Gtk::POLICY_AUTOMATIC; std::string property = xml_property (definition, "hscroll", styles, ""); if (property == "never") { hscrollbar_policy = Gtk::POLICY_NEVER; } else if (property == "always") { hscrollbar_policy = Gtk::POLICY_ALWAYS; } else if (property == "auto") { hscrollbar_policy = Gtk::POLICY_AUTOMATIC; } property = xml_property (definition, "vscroll", styles, ""); if (property == "never") { vscrollbar_policy = Gtk::POLICY_NEVER; } else if (property == "always") { vscrollbar_policy = Gtk::POLICY_ALWAYS; } else if (property == "auto") { vscrollbar_policy = Gtk::POLICY_AUTOMATIC; } ((Gtk::ScrolledWindow*)child)->set_policy(hscrollbar_policy, vscrollbar_policy); } else if (widget_type == "VBOX") { child = manage (new Gtk::VBox); } else if (widget_type == "HBOX") { child = manage (new Gtk::HBox); } if (child != NULL) { if (!widget_id.empty()) { named_widgets[widget_id] = child; } set_attributes(*child, definition, styles); } return child; } Gtk::Widget* WavesUI::add_widget (Gtk::Box& parent, const XMLNode& definition, const XMLNodeMap& styles, std::map& named_widgets) { Gtk::Widget* child = create_widget(definition, styles, named_widgets); if (child != NULL) { parent.pack_start(*child, false, false); } return child; } Gtk::Widget* WavesUI::add_widget (Gtk::ScrolledWindow& parent, const XMLNode& definition, const XMLNodeMap& styles, std::map& named_widgets) { Gtk::Widget* child = create_widget(definition, styles, named_widgets); if (child != NULL) { parent.add(*child); } return child; } Gtk::Widget* WavesUI::add_widget (Gtk::Layout& parent, const XMLNode& definition, const XMLNodeMap& styles, std::map& named_widgets) { Gtk::Widget* child = create_widget(definition, styles, named_widgets); if (child != NULL) { parent.put (*child, xml_property (definition, "x", styles, 0), xml_property (definition, "y", styles, 0)); } return child; } Gtk::Widget* WavesUI::add_widget (Gtk::Container& parent, const XMLNode& definition, const XMLNodeMap& styles, std::map& named_widgets) { Gtk::Widget* child = NULL; if(dynamic_cast (&parent)) { child = WavesUI::add_widget (*dynamic_cast (&parent), definition, styles, named_widgets); } else if(dynamic_cast (&parent)) { child = WavesUI::add_widget (*dynamic_cast (&parent), definition, styles, named_widgets); } else if(dynamic_cast (&parent)) { child = WavesUI::add_widget (*dynamic_cast (&parent), definition, styles, named_widgets); } Gtk::Container* container = dynamic_cast(child); if (container != NULL) { WavesUI::create_ui (definition.children(), styles, *container, named_widgets); Gtk::ScrolledWindow* sw = dynamic_cast(child); if (sw != NULL) { Gtk::Viewport* vp = (Gtk::Viewport*)sw->get_child(); if (vp != NULL) { set_attributes(*(Gtk::Widget*)vp, definition, styles); vp->set_shadow_type(Gtk::SHADOW_NONE); } } } return child; } void WavesUI::create_ui (const XMLNodeList& definition, const XMLNodeMap& styles, Gtk::Container& root, std::map& named_widgets) { for (XMLNodeList::const_iterator i = definition.begin(); i != definition.end(); ++i) { WavesUI::add_widget (root, **i, styles, named_widgets); } } void WavesUI::create_ui (const XMLTree& layout, Gtk::Container& root, std::map& named_widgets) { XMLNodeMap styles; get_styles(layout, styles); const XMLNodeList& definition = layout.root()->children(); WavesUI::create_ui (definition, styles, root, named_widgets); } static std::map xml_tree_cache; const XMLTree* WavesUI::load_layout (const std::string xml_file_name) { std::map::const_iterator it = xml_tree_cache.find(xml_file_name); if (it != xml_tree_cache.end()) { return (*it).second; } std::string layout_file; Searchpath spath (ardour_data_search_path()); spath.add_subdirectory_to_paths("ui"); if (!find_file_in_search_path (spath, xml_file_name, layout_file)) { dbg_msg("File not found: " + xml_file_name); return NULL; } const XMLTree* tree = new XMLTree (layout_file, false); xml_tree_cache[xml_file_name] = tree; return tree; } void WavesUI::set_attributes (Gtk::Widget& widget, const XMLNode& definition, const XMLNodeMap& styles) { int height = xml_property (definition, "height", styles, -1); int width = xml_property (definition, "width", styles, -1); widget.set_size_request (width, height); std::string property = xml_property (definition, "bgnormal", styles, ""); if (!property.empty ()) { widget.modify_bg(Gtk::STATE_NORMAL, Gdk::Color(property)); } property = xml_property (definition, "bgdisabled", styles, property); if (!property.empty ()) { widget.modify_bg(Gtk::STATE_INSENSITIVE, Gdk::Color(property)); } property = xml_property (definition, "bgactive", styles, ""); if (!property.empty ()) { widget.modify_bg(Gtk::STATE_ACTIVE, Gdk::Color(property)); } property = xml_property (definition, "bghover", styles, ""); if (!property.empty ()) { widget.modify_bg(Gtk::STATE_PRELIGHT, Gdk::Color(property)); } property = xml_property (definition, "fgnormal", styles, ""); if (!property.empty ()) { widget.modify_fg(Gtk::STATE_NORMAL, Gdk::Color(property)); } property = xml_property (definition, "fgdisabled", styles, property); if (!property.empty ()) { widget.modify_fg(Gtk::STATE_INSENSITIVE, Gdk::Color(property)); } property = xml_property (definition, "fgactive", styles, ""); if (!property.empty ()) { widget.modify_fg(Gtk::STATE_ACTIVE, Gdk::Color(property)); } property = xml_property (definition, "fghover", styles, ""); if (!property.empty ()) { widget.modify_fg(Gtk::STATE_PRELIGHT, Gdk::Color(property)); } property = xml_property (definition, "font", styles, ""); if (!property.empty ()) { widget.modify_font(Pango::FontDescription(property)); } if (xml_property (definition, "visible", styles, true)) { widget.show(); } else { widget.hide(); } Gtk::Box* box = dynamic_cast (&widget); if (box) { box->set_spacing(xml_property (definition, "spacing", styles, 0)); } } Gtk::Widget* WavesUI::WidgetMap::get_widget(char *id) { Gtk::Widget *child = NULL; std::map::iterator it = find(id); if(it != end()) child = it->second; return child; } Gtk::VBox& WavesUI::WidgetMap::get_vbox (char* id) { Gtk::VBox* child = dynamic_cast (get_widget(id)); if (child == NULL ) { throw std::exception(); } return *child; } Gtk::HBox& WavesUI::WidgetMap::get_hbox (char* id) { Gtk::HBox* child = dynamic_cast (get_widget(id)); if (child == NULL ) { throw std::exception(); } return *child; } Gtk::Layout& WavesUI::WidgetMap::get_layout (char* id) { Gtk::Layout* child = dynamic_cast (get_widget(id)); if (child == NULL ) { throw std::exception(); } return *child; } Gtk::Label& WavesUI::WidgetMap::get_label (char* id) { Gtk::Label* child = dynamic_cast (get_widget(id)); if (child == NULL ) { throw std::exception(); } return *child; } Gtk::ComboBoxText& WavesUI::WidgetMap::get_combo_box_text (char* id) { Gtk::ComboBoxText* child = dynamic_cast (get_widget(id)); if (child == NULL ) { throw std::exception(); } return *child; } WavesButton& WavesUI::WidgetMap::get_waves_button (char* id) { WavesButton* child = dynamic_cast (get_widget(id)); if (child == NULL ) { throw std::exception(); } return *child; }