/[svn]/gigedit/trunk/src/gigedit/mainwindow.cpp
ViewVC logotype

Diff of /gigedit/trunk/src/gigedit/mainwindow.cpp

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 3368 by schoenebeck, Thu Nov 16 19:18:42 2017 UTC revision 3419 by schoenebeck, Sat Feb 10 12:01:09 2018 UTC
# Line 457  MainWindow::MainWindow() : Line 457  MainWindow::MainWindow() :
457          m_actionGroup->add_action_bool("Statusbar", sigc::mem_fun(*this, &MainWindow::on_action_view_status_bar), true);          m_actionGroup->add_action_bool("Statusbar", sigc::mem_fun(*this, &MainWindow::on_action_view_status_bar), true);
458      m_actionToggleRestoreWinDim =      m_actionToggleRestoreWinDim =
459          m_actionGroup->add_action_bool("AutoRestoreWinDim", sigc::mem_fun(*this, &MainWindow::on_auto_restore_win_dim), Settings::singleton()->autoRestoreWindowDimension);          m_actionGroup->add_action_bool("AutoRestoreWinDim", sigc::mem_fun(*this, &MainWindow::on_auto_restore_win_dim), Settings::singleton()->autoRestoreWindowDimension);
460        m_actionToggleShowTooltips = m_actionGroup->add_action_bool(
461            "ShowTooltips", sigc::mem_fun(*this, &MainWindow::on_action_show_tooltips),
462            Settings::singleton()->showTooltips
463        );
464      m_actionToggleSaveWithTempFile =      m_actionToggleSaveWithTempFile =
465          m_actionGroup->add_action_bool("SaveWithTemporaryFile", sigc::mem_fun(*this, &MainWindow::on_save_with_temporary_file), Settings::singleton()->saveWithTemporaryFile);          m_actionGroup->add_action_bool("SaveWithTemporaryFile", sigc::mem_fun(*this, &MainWindow::on_save_with_temporary_file), Settings::singleton()->saveWithTemporaryFile);
466      m_actionGroup->add_action("RefreshAll", sigc::mem_fun(*this, &MainWindow::on_action_refresh_all));      m_actionGroup->add_action("RefreshAll", sigc::mem_fun(*this, &MainWindow::on_action_refresh_all));
# Line 480  MainWindow::MainWindow() : Line 484  MainWindow::MainWindow() :
484                           *this, &MainWindow::on_auto_restore_win_dim));                           *this, &MainWindow::on_auto_restore_win_dim));
485    
486      toggle_action =      toggle_action =
487            Gtk::ToggleAction::create("ShowTooltips", _("Tooltips for Beginners"));
488        toggle_action->set_active(Settings::singleton()->showTooltips);
489        actionGroup->add(
490            toggle_action,
491            sigc::mem_fun(*this, &MainWindow::on_action_show_tooltips)
492        );
493    
494        toggle_action =
495          Gtk::ToggleAction::create("SaveWithTemporaryFile", _("Save with _temporary file"));          Gtk::ToggleAction::create("SaveWithTemporaryFile", _("Save with _temporary file"));
496      toggle_action->set_active(Settings::singleton()->saveWithTemporaryFile);      toggle_action->set_active(Settings::singleton()->saveWithTemporaryFile);
497      actionGroup->add(toggle_action,      actionGroup->add(toggle_action,
# Line 921  MainWindow::MainWindow() : Line 933  MainWindow::MainWindow() :
933          "          <attribute name='label' translatable='yes'>Statusbar</attribute>"          "          <attribute name='label' translatable='yes'>Statusbar</attribute>"
934          "          <attribute name='action'>AppMenu.Statusbar</attribute>"          "          <attribute name='action'>AppMenu.Statusbar</attribute>"
935          "        </item>"          "        </item>"
936            "        <item id='ShowTooltips'>"
937            "          <attribute name='label' translatable='yes'>Tooltips for Beginners</attribute>"
938            "          <attribute name='action'>AppMenu.ShowTooltips</attribute>"
939            "        </item>"
940          "        <item id='AutoRestoreWinDim'>"          "        <item id='AutoRestoreWinDim'>"
941          "          <attribute name='label' translatable='yes'>Auto restore Window Dimensions</attribute>"          "          <attribute name='label' translatable='yes'>Auto restore Window Dimensions</attribute>"
942          "          <attribute name='action'>AppMenu.AutoRestoreWinDim</attribute>"          "          <attribute name='action'>AppMenu.AutoRestoreWinDim</attribute>"
# Line 1151  MainWindow::MainWindow() : Line 1167  MainWindow::MainWindow() :
1167          "    </menu>"          "    </menu>"
1168          "    <menu action='MenuView'>"          "    <menu action='MenuView'>"
1169          "      <menuitem action='Statusbar'/>"          "      <menuitem action='Statusbar'/>"
1170            "      <menuitem action='ShowTooltips'/>"
1171          "      <menuitem action='AutoRestoreWinDim'/>"          "      <menuitem action='AutoRestoreWinDim'/>"
1172          "      <separator/>"          "      <separator/>"
1173          "      <menuitem action='RefreshAll'/>"          "      <menuitem action='RefreshAll'/>"
# Line 1237  MainWindow::MainWindow() : Line 1254  MainWindow::MainWindow() :
1254      {      {
1255          Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(          Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
1256              uiManager->get_widget("/MenuBar/MenuEdit/CopySampleLoop"));              uiManager->get_widget("/MenuBar/MenuEdit/CopySampleLoop"));
1257          item->set_tooltip_text(_("Used when dragging a sample to a region's sample reference field. You may disable this for example if you want to replace an existing sample in a region with a new sample, but don't want that the region's current loop informations to be altered by this action."));          item->set_tooltip_text(_("Used when dragging a sample to a region's sample reference field. You may disable this for example if you want to replace an existing sample in a region with a new sample, but don't want that the region's current loop information to be altered by this action."));
1258      }      }
1259      {      {
1260          Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(          Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
# Line 1252  MainWindow::MainWindow() : Line 1269  MainWindow::MainWindow() :
1269      {      {
1270          Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(          Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
1271              uiManager->get_widget("/MenuBar/MenuSettings/MoveRootNoteWithRegionMoved"));              uiManager->get_widget("/MenuBar/MenuSettings/MoveRootNoteWithRegionMoved"));
1272          item->set_tooltip_text(_("If checked, and when a region is moved by dragging it around on the virtual keyboard, the keybord position dependent pitch will move exactly with the amount of semi tones the region was moved around."));          item->set_tooltip_text(_("If checked, and when a region is moved by dragging it around on the virtual keyboard, the keyboard position dependent pitch will move exactly with the amount of semi tones the region was moved around."));
1273      }      }
1274      {      {
1275          Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(          Gtk::MenuItem* item = dynamic_cast<Gtk::MenuItem*>(
# Line 1339  MainWindow::MainWindow() : Line 1356  MainWindow::MainWindow() :
1356      m_TreeView.set_model(m_refTreeModelFilter);      m_TreeView.set_model(m_refTreeModelFilter);
1357    
1358      m_TreeView.get_selection()->set_mode(Gtk::SELECTION_MULTIPLE);      m_TreeView.get_selection()->set_mode(Gtk::SELECTION_MULTIPLE);
1359      m_TreeView.set_tooltip_text(_("Right click here for actions on instruments & MIDI Rules. Drag & drop to change the order of instruments."));      m_TreeView.set_has_tooltip(true);
1360        m_TreeView.signal_query_tooltip().connect(
1361            sigc::mem_fun(*this, &MainWindow::onQueryTreeViewTooltip)
1362        );
1363      instrument_name_connection = m_refTreeModel->signal_row_changed().connect(      instrument_name_connection = m_refTreeModel->signal_row_changed().connect(
1364          sigc::mem_fun(*this, &MainWindow::instrument_name_changed)          sigc::mem_fun(*this, &MainWindow::instrument_name_changed)
1365      );      );
# Line 1580  MainWindow::MainWindow() : Line 1600  MainWindow::MainWindow() :
1600          Gtk::AccelMap::add_entry("<Macros>/macro_9", GDK_KEY_F10, noModifier);          Gtk::AccelMap::add_entry("<Macros>/macro_9", GDK_KEY_F10, noModifier);
1601          Gtk::AccelMap::add_entry("<Macros>/macro_10", GDK_KEY_F11, noModifier);          Gtk::AccelMap::add_entry("<Macros>/macro_10", GDK_KEY_F11, noModifier);
1602          Gtk::AccelMap::add_entry("<Macros>/macro_11", GDK_KEY_F12, noModifier);          Gtk::AccelMap::add_entry("<Macros>/macro_11", GDK_KEY_F12, noModifier);
1603            Gtk::AccelMap::add_entry("<Macros>/macro_12", GDK_KEY_F13, noModifier);
1604            Gtk::AccelMap::add_entry("<Macros>/macro_13", GDK_KEY_F14, noModifier);
1605            Gtk::AccelMap::add_entry("<Macros>/macro_14", GDK_KEY_F15, noModifier);
1606            Gtk::AccelMap::add_entry("<Macros>/macro_15", GDK_KEY_F16, noModifier);
1607            Gtk::AccelMap::add_entry("<Macros>/macro_16", GDK_KEY_F17, noModifier);
1608            Gtk::AccelMap::add_entry("<Macros>/macro_17", GDK_KEY_F18, noModifier);
1609            Gtk::AccelMap::add_entry("<Macros>/macro_18", GDK_KEY_F19, noModifier);
1610          Gtk::AccelMap::add_entry("<Macros>/SetupMacros", 'm', primaryModifierKey);          Gtk::AccelMap::add_entry("<Macros>/SetupMacros", 'm', primaryModifierKey);
1611    
1612          Glib::RefPtr<Gtk::AccelGroup> accelGroup = this->get_accel_group();          Glib::RefPtr<Gtk::AccelGroup> accelGroup = this->get_accel_group();
# Line 1602  MainWindow::MainWindow() : Line 1629  MainWindow::MainWindow() :
1629          Gtk::AccelMap::add_entry("<Scripts>/script_9", GDK_KEY_F10, Gdk::SHIFT_MASK);          Gtk::AccelMap::add_entry("<Scripts>/script_9", GDK_KEY_F10, Gdk::SHIFT_MASK);
1630          Gtk::AccelMap::add_entry("<Scripts>/script_10", GDK_KEY_F11, Gdk::SHIFT_MASK);          Gtk::AccelMap::add_entry("<Scripts>/script_10", GDK_KEY_F11, Gdk::SHIFT_MASK);
1631          Gtk::AccelMap::add_entry("<Scripts>/script_11", GDK_KEY_F12, Gdk::SHIFT_MASK);          Gtk::AccelMap::add_entry("<Scripts>/script_11", GDK_KEY_F12, Gdk::SHIFT_MASK);
1632            Gtk::AccelMap::add_entry("<Scripts>/script_12", GDK_KEY_F13, Gdk::SHIFT_MASK);
1633            Gtk::AccelMap::add_entry("<Scripts>/script_13", GDK_KEY_F14, Gdk::SHIFT_MASK);
1634            Gtk::AccelMap::add_entry("<Scripts>/script_14", GDK_KEY_F15, Gdk::SHIFT_MASK);
1635            Gtk::AccelMap::add_entry("<Scripts>/script_15", GDK_KEY_F16, Gdk::SHIFT_MASK);
1636            Gtk::AccelMap::add_entry("<Scripts>/script_16", GDK_KEY_F17, Gdk::SHIFT_MASK);
1637            Gtk::AccelMap::add_entry("<Scripts>/script_17", GDK_KEY_F18, Gdk::SHIFT_MASK);
1638            Gtk::AccelMap::add_entry("<Scripts>/script_18", GDK_KEY_F19, Gdk::SHIFT_MASK);
1639            Gtk::AccelMap::add_entry("<Scripts>/DropAllScriptSlots", GDK_KEY_BackSpace, Gdk::SHIFT_MASK);
1640    
1641          Glib::RefPtr<Gtk::AccelGroup> accelGroup = this->get_accel_group();          Glib::RefPtr<Gtk::AccelGroup> accelGroup = this->get_accel_group();
1642          assign_scripts_menu->set_accel_group(accelGroup);          assign_scripts_menu->set_accel_group(accelGroup);
1643      }      }
1644    
1645        on_show_tooltips_changed();
1646    
1647      Glib::signal_idle().connect_once(      Glib::signal_idle().connect_once(
1648          sigc::mem_fun(*this, &MainWindow::bringToFront),          sigc::mem_fun(*this, &MainWindow::bringToFront),
1649          200          200
# Line 2589  void MainWindow::on_action_warn_user_on_ Line 2626  void MainWindow::on_action_warn_user_on_
2626          !Settings::singleton()->warnUserOnExtensions;          !Settings::singleton()->warnUserOnExtensions;
2627  }  }
2628    
2629    void MainWindow::on_action_show_tooltips() {
2630        Settings::singleton()->showTooltips =
2631            !Settings::singleton()->showTooltips;
2632    
2633        on_show_tooltips_changed();
2634    }
2635    
2636    void MainWindow::on_show_tooltips_changed() {
2637        const bool b = Settings::singleton()->showTooltips;
2638    
2639        dimreg_label.set_has_tooltip(b);
2640        dimreg_all_regions.set_has_tooltip(b);
2641        dimreg_all_dimregs.set_has_tooltip(b);
2642        dimreg_stereo.set_has_tooltip(b);
2643    
2644        // Not doing this here, we let onQueryTreeViewTooltip() handle this per cell
2645        //m_TreeView.set_has_tooltip(b);
2646    
2647        m_TreeViewSamples.set_has_tooltip(b);
2648        m_TreeViewScripts.set_has_tooltip(b);
2649    
2650        set_has_tooltip(b);
2651    }
2652    
2653  void MainWindow::on_action_sync_sampler_instrument_selection() {  void MainWindow::on_action_sync_sampler_instrument_selection() {
2654      Settings::singleton()->syncSamplerInstrumentSelection =      Settings::singleton()->syncSamplerInstrumentSelection =
2655          !Settings::singleton()->syncSamplerInstrumentSelection;          !Settings::singleton()->syncSamplerInstrumentSelection;
# Line 2934  void MainWindow::updateSampleRefCountMap Line 2995  void MainWindow::updateSampleRefCountMap
2995      }      }
2996  }  }
2997    
2998    bool MainWindow::onQueryTreeViewTooltip(int x, int y, bool keyboardTip, const Glib::RefPtr<Gtk::Tooltip>& tooltip) {
2999        Gtk::TreeModel::iterator iter;
3000        m_TreeView.get_tooltip_context_iter(x, y, keyboardTip, iter);
3001        Gtk::TreeModel::Path path(iter);
3002        Gtk::TreeModel::Row row = *iter;
3003        Gtk::TreeViewColumn* pointedColumn = NULL;
3004        // resolve the precise table column the mouse points to
3005        {
3006            Gtk::TreeModel::Path path; // unused
3007            int cellX, cellY; // unused
3008            m_TreeView.get_path_at_pos(x, y, path, pointedColumn, cellX, cellY);
3009        }
3010        Gtk::TreeViewColumn* scriptsColumn = m_TreeView.get_column(2);
3011        if (pointedColumn == scriptsColumn) { // mouse hovers scripts column ...
3012            // show the script(s) assigned to the hovered instrument as tooltip
3013            tooltip->set_markup( row[m_Columns.m_col_tooltip] );
3014            m_TreeView.set_tooltip_cell(tooltip, &path, scriptsColumn, NULL);
3015        } else {
3016            // if beginners' tooltips is disabled then don't show the following one
3017            if (!Settings::singleton()->showTooltips)
3018                return false;
3019            // yeah, a beginners tooltip
3020            tooltip->set_text(_(
3021                "Right click here for actions on instruments & MIDI Rules. "
3022                "Drag & drop to change the order of instruments."
3023            ));
3024            m_TreeView.set_tooltip_cell(tooltip, &path, pointedColumn, NULL);
3025        }
3026        return true;
3027    }
3028    
3029    static Glib::ustring scriptTooltipFor(gig::Instrument* instrument, int index) {
3030        Glib::ustring name(gig_to_utf8(instrument->pInfo->Name));
3031        const int iScriptSlots = instrument->ScriptSlotCount();
3032        Glib::ustring tooltip = "<u>(" + ToString(index) + ") „"  + name + "”</u>\n\n";
3033        if (!iScriptSlots)
3034            tooltip += "<span foreground='red'><i>No script assigned</i></span>";
3035        else {
3036            for (int i = 0; i < iScriptSlots; ++i) {
3037                tooltip += "• " + ToString(i+1) + ". Script:  „<span foreground='#46DEFF'><b>" +
3038                           instrument->GetScriptOfSlot(i)->Name + "</b></span>”";
3039                if (i + 1 < iScriptSlots) tooltip += "\n\n";
3040            }
3041        }
3042        return tooltip;
3043    }
3044    
3045  void MainWindow::load_gig(gig::File* gig, const char* filename, bool isSharedInstrument)  void MainWindow::load_gig(gig::File* gig, const char* filename, bool isSharedInstrument)
3046  {  {
3047      file = 0;      file = 0;
# Line 2963  void MainWindow::load_gig(gig::File* gig Line 3071  void MainWindow::load_gig(gig::File* gig
3071          row[m_Columns.m_col_name] = name;          row[m_Columns.m_col_name] = name;
3072          row[m_Columns.m_col_instr] = instrument;          row[m_Columns.m_col_instr] = instrument;
3073          row[m_Columns.m_col_scripts] = iScriptSlots ? ToString(iScriptSlots) : "";          row[m_Columns.m_col_scripts] = iScriptSlots ? ToString(iScriptSlots) : "";
3074            row[m_Columns.m_col_tooltip] = scriptTooltipFor(instrument, index);
3075    
3076  #if !USE_GTKMM_BUILDER  #if !USE_GTKMM_BUILDER
3077          add_instrument_to_menu(name);          add_instrument_to_menu(name);
# Line 3080  void MainWindow::instr_name_changed_by_i Line 3189  void MainWindow::instr_name_changed_by_i
3189      gig::Instrument* instrument = row[m_Columns.m_col_instr];      gig::Instrument* instrument = row[m_Columns.m_col_instr];
3190      Glib::ustring gigname(gig_to_utf8(instrument->pInfo->Name));      Glib::ustring gigname(gig_to_utf8(instrument->pInfo->Name));
3191      if (gigname != name) {      if (gigname != name) {
3192            Gtk::TreeModel::Path path(*it);
3193            const int index = path[0];
3194          row[m_Columns.m_col_name] = gigname;          row[m_Columns.m_col_name] = gigname;
3195            row[m_Columns.m_col_tooltip] = scriptTooltipFor(instrument, index);
3196      }      }
3197  }  }
3198    
# Line 3126  void MainWindow::onScriptSlotsModified(g Line 3238  void MainWindow::onScriptSlotsModified(g
3238          Gtk::TreeModel::Row row = model->children()[i];          Gtk::TreeModel::Row row = model->children()[i];
3239          if (row[m_Columns.m_col_instr] != pInstrument) continue;          if (row[m_Columns.m_col_instr] != pInstrument) continue;
3240          row[m_Columns.m_col_scripts] = iScriptSlots ? ToString(iScriptSlots) : "";          row[m_Columns.m_col_scripts] = iScriptSlots ? ToString(iScriptSlots) : "";
3241            row[m_Columns.m_col_tooltip] = scriptTooltipFor(pInstrument, i);
3242          break;          break;
3243      }      }
3244    
# Line 3151  void MainWindow::assignScript(gig::Scrip Line 3264  void MainWindow::assignScript(gig::Scrip
3264      onScriptSlotsModified(pInstrument);      onScriptSlotsModified(pInstrument);
3265  }  }
3266    
3267    void MainWindow::dropAllScriptSlots() {
3268        gig::Instrument* pInstrument = get_instrument();
3269        if (!pInstrument) {
3270            printf("!instrument\n");
3271            return;
3272        }
3273    
3274        const int iScriptSlots = pInstrument->ScriptSlotCount();
3275        for (int i = iScriptSlots - 1; i >= 0; --i)
3276            pInstrument->RemoveScriptSlot(i);
3277    
3278        onScriptSlotsModified(pInstrument);
3279    }
3280    
3281  void MainWindow::on_action_refresh_all() {  void MainWindow::on_action_refresh_all() {
3282      __refreshEntireGUI();      __refreshEntireGUI();
3283  }  }
# Line 3591  void MainWindow::updateScriptListOfMenu( Line 3718  void MainWindow::updateScriptListOfMenu(
3718          assign_scripts_menu->append(*item);          assign_scripts_menu->append(*item);
3719      }      }
3720    
3721        // add separator line to menu
3722        assign_scripts_menu->append(*new Gtk::SeparatorMenuItem);
3723    
3724        {
3725            Gtk::MenuItem* item = new Gtk::MenuItem(_("Unassign All Scripts"));
3726            item->signal_activate().connect(
3727                sigc::mem_fun(*this, &MainWindow::dropAllScriptSlots)
3728            );
3729            assign_scripts_menu->append(*item);
3730            item->set_accel_path("<Scripts>/DropAllScriptSlots");
3731        }
3732    
3733  #if HAS_GTKMM_SHOW_ALL_CHILDREN  #if HAS_GTKMM_SHOW_ALL_CHILDREN
3734      assign_scripts_menu->show_all_children();      assign_scripts_menu->show_all_children();
3735  #endif  #endif
# Line 3639  void MainWindow::add_instrument(gig::Ins Line 3778  void MainWindow::add_instrument(gig::Ins
3778      instrument_name_connection.block();      instrument_name_connection.block();
3779      Gtk::TreeModel::iterator iterInstr = m_refTreeModel->append();      Gtk::TreeModel::iterator iterInstr = m_refTreeModel->append();
3780      Gtk::TreeModel::Row rowInstr = *iterInstr;      Gtk::TreeModel::Row rowInstr = *iterInstr;
3781      rowInstr[m_Columns.m_col_nr] = m_refTreeModel->children().size() - 1;      const int index = m_refTreeModel->children().size() - 1;
3782        rowInstr[m_Columns.m_col_nr] = index;
3783      rowInstr[m_Columns.m_col_name] = name;      rowInstr[m_Columns.m_col_name] = name;
3784      rowInstr[m_Columns.m_col_instr] = instrument;      rowInstr[m_Columns.m_col_instr] = instrument;
3785      rowInstr[m_Columns.m_col_scripts] = "";      rowInstr[m_Columns.m_col_scripts] = "";
3786        rowInstr[m_Columns.m_col_tooltip] = scriptTooltipFor(instrument, index);
3787      instrument_name_connection.unblock();      instrument_name_connection.unblock();
3788    
3789  #if !USE_GTKMM_BUILDER  #if !USE_GTKMM_BUILDER
# Line 3729  void MainWindow::on_action_remove_instru Line 3870  void MainWindow::on_action_remove_instru
3870                       it != m_refTreeModel->children().end(); ++it, ++index)                       it != m_refTreeModel->children().end(); ++it, ++index)
3871                  {                  {
3872                      Gtk::TreeModel::Row row = *it;                      Gtk::TreeModel::Row row = *it;
3873                        gig::Instrument* instrument = row[m_Columns.m_col_instr];
3874                      row[m_Columns.m_col_nr] = index;                      row[m_Columns.m_col_nr] = index;
3875                        row[m_Columns.m_col_tooltip] = scriptTooltipFor(instrument, index);
3876                  }                  }
3877              }              }
3878    
# Line 3988  void MainWindow::add_or_replace_sample(b Line 4131  void MainWindow::add_or_replace_sample(b
4131          for (std::vector<std::string>::iterator iter = filenames.begin();          for (std::vector<std::string>::iterator iter = filenames.begin();
4132               iter != filenames.end(); ++iter) {               iter != filenames.end(); ++iter) {
4133              printf("Adding sample %s\n",(*iter).c_str());              printf("Adding sample %s\n",(*iter).c_str());
4134              // use libsndfile to retrieve file informations              // use libsndfile to retrieve file information
4135              SF_INFO info;              SF_INFO info;
4136              info.format = 0;              info.format = 0;
4137              SNDFILE* hFile = sf_open((*iter).c_str(), SFM_READ, &info);              SNDFILE* hFile = sf_open((*iter).c_str(), SFM_READ, &info);

Legend:
Removed from v.3368  
changed lines
  Added in v.3419

  ViewVC Help
Powered by ViewVC