Rivendellaudio/rdcatch/rdcatch.cpp

2593 lines
76 KiB
C++

// rdcatch.cpp
//
// The Event Schedule Manager for Rivendell.
//
// (C) Copyright 2002-2006 Fred Gleason <fredg@paravelsystems.com>
//
// $Id: rdcatch.cpp,v 1.127.4.8 2014/02/11 23:46:30 cvs Exp $
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License version 2 as
// published by the Free Software Foundation.
//
// 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 <unistd.h>
#include <vector>
#include <qapplication.h>
#include <qwindowsstyle.h>
#include <qwidget.h>
#include <qpainter.h>
#include <qsqlpropertymap.h>
#include <qmessagebox.h>
#include <qpushbutton.h>
#include <qlabel.h>
#include <qlabel.h>
#include <qlistview.h>
#include <qsignalmapper.h>
#include <qtextcodec.h>
#include <qtranslator.h>
#include <qlayout.h>
#include <rdprofile.h>
#include <rddb.h>
#include <rdconf.h>
#include <rd.h>
#include <rduser.h>
#include <rdripc.h>
#include <rdcut.h>
#include <rdcatch.h>
#include <rdstation.h>
#include <rddeck.h>
#include <rdaudio_port.h>
#include <rdcut_path.h>
#include <rdmixer.h>
#include <rdcheck_daemons.h>
#include <rdsettings.h>
#include <rdcmd_switch.h>
#include <rdedit_audio.h>
#include <dbversion.h>
#include <add_recording.h>
#include <edit_recording.h>
#include <edit_playout.h>
#include <edit_cartevent.h>
#include <edit_switchevent.h>
#include <edit_download.h>
#include <edit_upload.h>
#include <list_reports.h>
#include <deckmon.h>
#include <colors.h>
#include <globals.h>
#include <assert.h>
//
// Global Resources
//
RDConfig *catch_config;
RDStation *rdstation_conf;
RDAudioPort *rdaudioport_conf;
RDUser *catch_user;
RDLibraryConf *rdlibrary_conf;
RDRipc *catch_ripc;
RDCae *catch_cae;
RDCartDialog *catch_cart_dialog;
int catch_audition_card=-1;
int catch_audition_port=-1;
RDSystem *catch_system=NULL;
//
// Icons
//
#include "../icons/record.xpm"
#include "../icons/play.xpm"
#include "../icons/rml5.xpm"
#include "../icons/switch3.xpm"
#include "../icons/download.xpm"
#include "../icons/upload.xpm"
#include "../icons/rivendell-22x22.xpm"
MainWidget::MainWidget(QWidget *parent,const char *name)
:QWidget(parent,name)
{
QString str;
catch_host_warnings=false;
catch_audition_stream=-1;
bool skip_db_check=false;
unsigned schema=0;
catch_scroll=false;
//
// Read Command Options
//
RDCmdSwitch *cmd=new RDCmdSwitch(qApp->argc(),qApp->argv(),"rdcatch",
RDCATCH_USAGE);
for(unsigned i=0;i<cmd->keys();i++) {
if(cmd->key(i)=="--offline-host-warnings") {
catch_host_warnings=RDBool(cmd->value(i));
}
if(cmd->key(i)=="--skip-db-check") {
skip_db_check=true;
}
}
delete cmd;
//
// Fix the Window Size
//
setMinimumWidth(sizeHint().width());
setMinimumHeight(sizeHint().height());
//
// Generate Fonts
//
QFont default_font("Helvetica",12,QFont::Normal);
default_font.setPixelSize(12);
qApp->setFont(default_font);
QFont list_font=QFont("Helvetica",12,QFont::Normal);
list_font.setPixelSize(12);
QFont label_font=QFont("Helvetica",10,QFont::Bold);
label_font.setPixelSize(10);
QFont button_font=QFont("Helvetica",12,QFont::Bold);
button_font.setPixelSize(12);
QFont clock_font=QFont("Helvetica",18,QFont::Bold);
clock_font.setPixelSize(18);
//
// Create Icons
//
catch_record_map=new QPixmap(record_xpm);
catch_playout_map=new QPixmap(play_xpm);
catch_macro_map=new QPixmap(rml5_xpm);
catch_switch_map=new QPixmap(switch3_xpm);
catch_download_map=new QPixmap(download_xpm);
catch_upload_map=new QPixmap(upload_xpm);
catch_rivendell_map=new QPixmap(rivendell_xpm);
setIcon(*catch_rivendell_map);
//
// Generate Palettes
//
catch_scroll_color[0]=palette();
catch_scroll_color[0].setColor(QPalette::Active,QColorGroup::ButtonText,
BUTTON_ACTIVE_TEXT_COLOR);
catch_scroll_color[0].setColor(QPalette::Active,QColorGroup::Button,
BUTTON_ACTIVE_BACKGROUND_COLOR);
catch_scroll_color[0].setColor(QPalette::Active,QColorGroup::Background,
backgroundColor());
catch_scroll_color[0].setColor(QPalette::Inactive,QColorGroup::ButtonText,
BUTTON_ACTIVE_TEXT_COLOR);
catch_scroll_color[0].setColor(QPalette::Inactive,QColorGroup::Button,
BUTTON_ACTIVE_BACKGROUND_COLOR);
catch_scroll_color[0].setColor(QPalette::Inactive,QColorGroup::Background,
backgroundColor());
catch_scroll_color[1]=QPalette(backgroundColor(),backgroundColor());
//
// Ensure the system daemons are running
//
RDInitializeDaemons();
//
// Load Local Configs
//
catch_config=new RDConfig();
catch_config->load();
str=QString("RDCatch")+" v"+VERSION+" - "+tr("Host")+":";
setCaption(QString().sprintf("%s %s",(const char *)str,
(const char *)catch_config->stationName()));
//
// Open Database
//
QString err (tr("rdcatch : "));
catch_db=RDInitDb(&schema,&err);
if(!catch_db) {
log(RDConfig::LogErr,err);
exit(0);
}
if((schema!=RD_VERSION_DATABASE)&&(!skip_db_check)) {
fprintf(stderr,"rdcatch: database version mismatch, should be %u, is %u\n",
RD_VERSION_DATABASE,schema);
exit(256);
}
connect(RDDbStatus(),SIGNAL(logText(RDConfig::LogPriority,const QString &)),
this,SLOT(log(RDConfig::LogPriority,const QString &)));
//
// Allocate Global Resources
//
rdstation_conf=new RDStation(catch_config->stationName());
catch_audition_card=rdstation_conf->cueCard();
catch_audition_port=rdstation_conf->cuePort();
catch_time_offset=rdstation_conf->timeOffset();
catch_system=new RDSystem();
//
// Load Audio Settings
//
RDDeck *deck=new RDDeck(catch_config->stationName(),0);
delete deck;
head_playing=false;
tail_playing=false;
rdaudioport_conf=new RDAudioPort(rdstation_conf->name(),catch_audition_card);
//
// Library Config
//
rdlibrary_conf=new RDLibraryConf(catch_config->stationName(),0);
//
// RIPC Connection
//
catch_ripc=new RDRipc(catch_config->stationName());
connect(catch_ripc,SIGNAL(connected(bool)),
this,SLOT(ripcConnectedData(bool)));
catch_user=NULL;
connect(catch_ripc,SIGNAL(userChanged()),this,SLOT(ripcUserData()));
catch_ripc->connectHost("localhost",RIPCD_TCP_PORT,catch_config->password());
//
// CAE Connection
//
catch_cae=new RDCae(rdstation_conf,catch_config,this,"catch_cae");
connect(catch_cae,SIGNAL(isConnected(bool)),this,SLOT(initData(bool)));
connect(catch_cae,SIGNAL(playing(int)),this,SLOT(playedData(int)));
connect(catch_cae,SIGNAL(playStopped(int)),
this,SLOT(playStoppedData(int)));
catch_cae->connectHost();
//
// Set Audio Assignments
//
RDSetMixerPorts(rdstation_conf->name(),catch_cae);
//
// Deck Monitors
//
catch_monitor_view=new QScrollView(this,"catch_monitor_view",
Qt::WNoAutoErase);
catch_monitor_vbox=new VBox(catch_monitor_view);
catch_monitor_vbox->setSpacing(2);
catch_monitor_view->addChild(catch_monitor_vbox);
QSignalMapper *mapper=new QSignalMapper(this,"deck_mapper");
connect(mapper,SIGNAL(mapped(int)),this,SLOT(abortData(int)));
QSignalMapper *mon_mapper=new QSignalMapper(this,"monitor_mapper");
connect(mon_mapper,SIGNAL(mapped(int)),this,SLOT(monitorData(int)));
QString sql;
RDSqlQuery *q1;
RDSqlQuery *q=
new RDSqlQuery("select NAME,IPV4_ADDRESS from STATIONS\
where NAME!=\"DEFAULT\"");
catch_station_count=0;
while(q->next()) {
catch_connect[catch_station_count].connect=
new RDCatchConnect(catch_station_count,this,"catch_connect");
catch_connect[catch_station_count].station=
q->value(0).toString().lower();
connect(catch_connect[catch_station_count].connect,
SIGNAL(statusChanged(int,unsigned,RDDeck::Status,int,const QString &)),
this,
SLOT(statusChangedData(int,unsigned,RDDeck::Status,int,const QString &)));
connect(catch_connect[catch_station_count].connect,
SIGNAL(monitorChanged(int,unsigned,bool)),
this,SLOT(monitorChangedData(int,unsigned,bool)));
connect(catch_connect[catch_station_count].connect,
SIGNAL(connected(int,bool)),
this,SLOT(connectedData(int,bool)));
connect(catch_connect[catch_station_count].connect,
SIGNAL(meterLevel(int,int,int,int)),
this,SLOT(meterLevelData(int,int,int,int)));
connect(catch_connect[catch_station_count].connect,
SIGNAL(eventUpdated(int)),
this,SLOT(eventUpdatedData(int)));
connect(catch_connect[catch_station_count].connect,
SIGNAL(eventPurged(int)),
this,SLOT(eventPurgedData(int)));
connect(catch_connect[catch_station_count].connect,
SIGNAL(heartbeatFailed(int)),
this,SLOT(heartbeatFailedData(int)));
catch_connect[catch_station_count].connect->
connectHost(q->value(1).toString(),RDCATCHD_TCP_PORT,
catch_config->password());
catch_station_count++;
sql=QString().sprintf("select CHANNEL,MON_PORT_NUMBER from DECKS \
where (CARD_NUMBER!=-1)&&(PORT_NUMBER!=-1)&&(CHANNEL>0)&&(STATION_NAME=\"%s\") \
order by CHANNEL",(const char *)q->value(0).toString().lower());
q1=new RDSqlQuery(sql);
while(q1->next()) {
catch_connect[catch_station_count-1].
chan.push_back(q1->value(0).toUInt());
catch_connect[catch_station_count-1].
mon_id.push_back(catch_monitor.size());
catch_monitor.push_back(new CatchMonitor());
catch_monitor.back()->setDeckMon(new DeckMon(q->value(0).toString(),
q1->value(0).toUInt(),
catch_monitor_vbox));
catch_monitor.back()->setSerialNumber(catch_station_count-1);
catch_monitor.back()->setChannelNumber(q1->value(0).toUInt());
catch_monitor_vbox->addWidget(catch_monitor.back()->deckMon());
catch_monitor.back()->deckMon()->
enableMonitorButton((q1->value(1).toInt()>=0)&&
(catch_config->stationName().lower()==
q->value(0).toString().lower()));
catch_monitor.back()->deckMon()->show();
mapper->setMapping(catch_monitor.back()->deckMon(),
catch_monitor.size()-1);
connect(catch_monitor.back()->deckMon(),SIGNAL(abortClicked()),
mapper,SLOT(map()));
mon_mapper->setMapping(catch_monitor.back()->deckMon(),
catch_monitor.size()-1);
connect(catch_monitor.back()->deckMon(),SIGNAL(monitorClicked()),
mon_mapper,SLOT(map()));
}
delete q1;
}
delete q;
if(catch_monitor.size()==0) {
catch_monitor_view->hide();
}
//
// User
//
catch_user=NULL;
//
// Filter Selectors
//
catch_show_active_box=new QCheckBox(this,"catch_show_active_box");
catch_show_active_label=new QLabel(catch_show_active_box,
tr("Show Only Active Events"),
this,"catch_show_active_label");
catch_show_active_label->setFont(label_font);
catch_show_active_label->setAlignment(AlignLeft|AlignVCenter);
connect(catch_show_active_box,SIGNAL(toggled(bool)),
this,SLOT(filterChangedData(bool)));
catch_show_today_box=new QCheckBox(this,"catch_show_today_box");
catch_show_today_label=new QLabel(catch_show_active_box,
tr("Show Only Today's Events"),
this,"catch_show_today_label");
catch_show_today_label->setFont(label_font);
catch_show_today_label->setAlignment(AlignLeft|AlignVCenter);
connect(catch_show_today_box,SIGNAL(toggled(bool)),
this,SLOT(filterChangedData(bool)));
catch_dow_box=new QComboBox(this,"catch_down_box");
catch_dow_label=new QLabel(catch_dow_box,tr("Show DayOfWeek:"),
this,"catch_dow_label");
catch_dow_label->setFont(label_font);
catch_dow_label->setAlignment(AlignRight|AlignVCenter);
catch_dow_box->insertItem(tr("All"));
catch_dow_box->insertItem(tr("Weekdays"));
catch_dow_box->insertItem(tr("Sunday"));
catch_dow_box->insertItem(tr("Monday"));
catch_dow_box->insertItem(tr("Tuesday"));
catch_dow_box->insertItem(tr("Wednesday"));
catch_dow_box->insertItem(tr("Thursday"));
catch_dow_box->insertItem(tr("Friday"));
catch_dow_box->insertItem(tr("Saturday"));
connect(catch_dow_box,SIGNAL(activated(int)),this,SLOT(dowChangedData(int)));
//
// Cart Picker
//
catch_cart_dialog=new RDCartDialog(&catch_filter,&catch_group,
&catch_schedcode,catch_cae,catch_ripc,
rdstation_conf,catch_system,catch_config,
this);
//
// Cart List
//
catch_recordings_list=new CatchListView(this,"catch_recordings_list");
catch_recordings_list->setAllColumnsShowFocus(true);
catch_recordings_list->setItemMargin(5);
catch_recordings_list->setFont(list_font);
connect(catch_recordings_list,SIGNAL(selectionChanged(QListViewItem *)),
this,SLOT(selectionChangedData(QListViewItem *)));
connect(catch_recordings_list,
SIGNAL(doubleClicked(QListViewItem *,const QPoint &,int)),
this,
SLOT(doubleClickedData(QListViewItem *,const QPoint &,int)));
catch_recordings_list->addColumn("");
catch_recordings_list->setColumnAlignment(0,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("DESCRIPTION"));
catch_recordings_list->setColumnAlignment(1,Qt::AlignLeft);
catch_recordings_list->addColumn(tr("LOCATION"));
catch_recordings_list->setColumnAlignment(2,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("START"));
catch_recordings_list->setColumnAlignment(3,Qt::AlignLeft);
catch_recordings_list->addColumn(tr("END"));
catch_recordings_list->setColumnAlignment(4,Qt::AlignLeft);
catch_recordings_list->addColumn(tr("SOURCE"));
catch_recordings_list->setColumnAlignment(5,Qt::AlignLeft);
catch_recordings_list->addColumn(tr("DESTINATION"));
catch_recordings_list->setColumnAlignment(6,Qt::AlignLeft);
catch_recordings_list->addColumn("");
catch_recordings_list->setColumnAlignment(7,Qt::AlignHCenter);
catch_recordings_list->addColumn("");
catch_recordings_list->setColumnAlignment(8,Qt::AlignHCenter);
catch_recordings_list->addColumn("");
catch_recordings_list->setColumnAlignment(9,Qt::AlignHCenter);
catch_recordings_list->addColumn("");
catch_recordings_list->setColumnAlignment(10,Qt::AlignHCenter);
catch_recordings_list->addColumn("");
catch_recordings_list->setColumnAlignment(11,Qt::AlignHCenter);
catch_recordings_list->addColumn("");
catch_recordings_list->setColumnAlignment(12,Qt::AlignHCenter);
catch_recordings_list->addColumn("");
catch_recordings_list->setColumnAlignment(13,Qt::AlignHCenter);
catch_recordings_list->addColumn("RSS FEED");
catch_recordings_list->setColumnAlignment(14,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("ORIGIN"));
catch_recordings_list->setColumnAlignment(15,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("ONE SHOT"));
catch_recordings_list->setColumnAlignment(16,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("TRIM THRESHOLD"));
catch_recordings_list->setColumnAlignment(17,Qt::AlignRight);
catch_recordings_list->addColumn(tr("STARTDATE OFFSET"));
catch_recordings_list->setColumnAlignment(18,Qt::AlignRight);
catch_recordings_list->addColumn(tr("ENDDATE OFFSET"));
catch_recordings_list->setColumnAlignment(19,Qt::AlignRight);
catch_recordings_list->addColumn(tr("FORMAT"));
catch_recordings_list->setColumnAlignment(20,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("CHANNELS"));
catch_recordings_list->setColumnAlignment(21,Qt::AlignRight);
catch_recordings_list->addColumn(tr("SAMPLE RATE"));
catch_recordings_list->setColumnAlignment(22,Qt::AlignRight);
catch_recordings_list->addColumn(tr("BIT RATE"));
catch_recordings_list->setColumnAlignment(23,Qt::AlignRight);
catch_recordings_list->addColumn(tr("STATION"));
catch_recordings_list->setColumnAlignment(24,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("DECK"));
catch_recordings_list->setColumnAlignment(25,Qt::AlignRight);
catch_recordings_list->addColumn(tr("CUT"));
catch_recordings_list->setColumnAlignment(26,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("CART"));
catch_recordings_list->setColumnAlignment(27,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("ID"));
catch_recordings_list->setColumnAlignment(28,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("TYPE"));
catch_recordings_list->setColumnAlignment(29,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("STATUS"));
catch_recordings_list->setColumnAlignment(30,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("EXIT CODE"));
catch_recordings_list->setColumnAlignment(31,Qt::AlignHCenter);
catch_recordings_list->addColumn(tr("STATE"));
catch_recordings_list->setColumnAlignment(32,Qt::AlignHCenter);
catch_recordings_list->setSorting(3); // Start Time
//
// Add Button
//
catch_add_button=new QPushButton(this,"add_button");
catch_add_button->setFont(button_font);
catch_add_button->setText(tr("&Add"));
connect(catch_add_button,SIGNAL(clicked()),this,SLOT(addData()));
//
// Edit Button
//
catch_edit_button=new QPushButton(this,"edit_button");
catch_edit_button->setFont(button_font);
catch_edit_button->setText(tr("&Edit"));
connect(catch_edit_button,SIGNAL(clicked()),this,SLOT(editData()));
//
// Delete Button
//
catch_delete_button=new QPushButton(this,"delete_button");
catch_delete_button->setFont(button_font);
catch_delete_button->setText(tr("&Delete"));
connect(catch_delete_button,SIGNAL(clicked()),this,SLOT(deleteData()));
//
// Scroll Button
//
catch_scroll_button=new QPushButton(this,"catch_scroll_button");
catch_scroll_button->setFont(button_font);
catch_scroll_button->setText(tr("Scroll"));
connect(catch_scroll_button,SIGNAL(clicked()),this,SLOT(scrollButtonData()));
//
// Reports Button
//
catch_reports_button=new QPushButton(this,"catch_reports_button");
catch_reports_button->setFont(button_font);
catch_reports_button->setText(tr("Reports"));
connect(catch_reports_button,SIGNAL(clicked()),this,SLOT(reportsButtonData()));
//
// Wall Clock
//
catch_clock_label=new QLabel("00:00:00",this,"catch_clock_label");
catch_clock_label->setFont(clock_font);
catch_clock_label->setAlignment(AlignCenter);
catch_clock_timer=new QTimer(this,"catch_clock_timer");
connect(catch_clock_timer,SIGNAL(timeout()),this,SLOT(clockData()));
clockData();
//
// Play Head Button
//
catch_head_button=
new RDTransportButton(RDTransportButton::PlayFrom,this,"catch_head_button");
catch_head_button->setDisabled(true);
connect(catch_head_button,SIGNAL(clicked()),this,SLOT(headButtonData()));
//
// Play Tail Button
//
catch_tail_button=
new RDTransportButton(RDTransportButton::PlayTo,this,"catch_tail_button");
catch_tail_button->setDisabled(true);
connect(catch_tail_button,SIGNAL(clicked()),this,SLOT(tailButtonData()));
//
// Play Stop Button
//
catch_stop_button=
new RDTransportButton(RDTransportButton::Stop,this,"catch_stop_button");
catch_stop_button->setDisabled(true);
catch_stop_button->setOnColor(red);
connect(catch_stop_button,SIGNAL(clicked()),this,SLOT(stopButtonData()));
catch_stop_button->on();
//
// Close Button
//
catch_close_button=new QPushButton(this,"close_button");
catch_close_button->setFont(button_font);
catch_close_button->setText(tr("&Close"));
catch_close_button->setFocus();
catch_close_button->setDefault(true);
connect(catch_close_button,SIGNAL(clicked()),this,SLOT(quitMainWidget()));
//
// Next Event Timer
//
catch_next_timer=new QTimer(this,"catch_next_timer");
connect(catch_next_timer,SIGNAL(timeout()),this,SLOT(nextEventData()));
//
// Midnight Timer
//
catch_midnight_timer=new QTimer(this,"catch_midnight_timer");
connect(catch_midnight_timer,SIGNAL(timeout()),this,SLOT(midnightData()));
midnightData();
LoadGeometry();
RefreshList();
QTime current_time=QTime::currentTime().addMSecs(catch_time_offset);
QDate current_date=QDate::currentDate();
QTime next_time;
if(ShowNextEvents(current_date.dayOfWeek(),current_time,&next_time)>0) {
catch_next_timer->start(current_time.msecsTo(next_time),true);
}
nextEventData();
//
// Silly Resize Workaround
// (so that the deck monitors get laid out properly)
//
QTimer *timer=new QTimer(this,"resize_timer");
connect(timer,SIGNAL(timeout()),this,SLOT(resizeData()));
timer->start(1,true);
}
void MainWidget::log(RDConfig::LogPriority prio,const QString &msg)
{
catch_config->log("RDCatch",prio,msg);
}
QSize MainWidget::sizeHint() const
{
return QSize(940,600);
}
QSizePolicy MainWidget::sizePolicy() const
{
return QSizePolicy(QSizePolicy::Fixed,QSizePolicy::Fixed);
}
void MainWidget::resizeData()
{
QResizeEvent *e=new QResizeEvent(QSize(geometry().width(),
geometry().height()),
QSize(geometry().width(),
geometry().height()));
resizeEvent(e);
delete e;
}
void MainWidget::connectedData(int serial,bool state)
{
if(state) {
catch_connect[serial].connect->enableMetering(true);
}
}
void MainWidget::nextEventData()
{
QTime next_time;
RDListViewItem *item=(RDListViewItem *)catch_recordings_list->firstChild();
if(item!=NULL) {
do {
if(item->backgroundColor()==EVENT_NEXT_COLOR) {
item->setBackgroundColor(catch_recordings_list->palette().color(QPalette::Active,QColorGroup::Base));
}
} while((item=(RDListViewItem *)item->nextSibling())!=NULL);
}
QTime current_time=QTime::currentTime().addMSecs(catch_time_offset);
QDate current_date=QDate::currentDate();
if(ShowNextEvents(current_date.dayOfWeek(),current_time,&next_time)>0) {
catch_next_timer->start(current_time.msecsTo(next_time),true);
if(catch_scroll) {
UpdateScroll();
}
return;
}
int extra_day=0;
for(int i=current_date.dayOfWeek()+1;i<8;i++) {
if(ShowNextEvents(i,QTime(),&next_time)>0) {
int interval=current_time.msecsTo(QTime(23,59,59))+1000+
86400000*extra_day+
QTime().msecsTo(next_time);
catch_next_timer->start(interval,true);
if(catch_scroll) {
UpdateScroll();
}
return;
}
extra_day++;
}
for(int i=1;i<(current_date.dayOfWeek()+1);i++) {
if(ShowNextEvents(i,QTime(),&next_time)>0) {
int interval=current_time.msecsTo(QTime(23,59,59))+1000+
86400000*extra_day+
QTime().msecsTo(next_time);
catch_next_timer->start(interval,true);
if(catch_scroll) {
UpdateScroll();
}
return;
}
extra_day++;
}
}
void MainWidget::addData()
{
RDSqlQuery *q;
RDListViewItem *item;
int conn;
if(!catch_user->editCatches()) {
return;
}
EnableScroll(false);
int n=AddRecord();
AddRecording *recording=new AddRecording(n,&catch_filter,this);
switch((RDRecording::Type)recording->exec()) {
case RDRecording::Recording:
case RDRecording::Playout:
case RDRecording::MacroEvent:
case RDRecording::SwitchEvent:
case RDRecording::Download:
case RDRecording::Upload:
item=new RDListViewItem(catch_recordings_list);
item->setBackgroundColor(catch_recordings_list->palette().color(QPalette::Active,QColorGroup::Base));
item->setText(28,QString().sprintf("%d",n));
RefreshLine(item);
conn=GetConnection(item->text(24));
if(conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
catch_recordings_list->setSelected(item,true);
catch_recordings_list->ensureItemVisible(item);
catch_connect[conn].connect->addEvent(n);
nextEventData();
break;
default:
q=new RDSqlQuery(QString().
sprintf("delete from RECORDINGS where ID=%d",n));
delete q;
break;
}
delete recording;
}
void MainWidget::editData()
{
int old_conn;
int new_conn;
std::vector<int> new_events;
RDListViewItem *item=(RDListViewItem *)catch_recordings_list->selectedItem();
EditRecording *recording;
EditPlayout *playout;
EditCartEvent *event;
EditSwitchEvent *switch_event;
EditDownload *download;
EditUpload *upload;
if(!catch_user->editCatches()) {
return;
}
if(item==NULL) {
return;
}
switch((RDRecording::ExitCode)item->text(31).toUInt()) {
case RDRecording::Downloading:
case RDRecording::Uploading:
case RDRecording::RecordActive:
case RDRecording::PlayActive:
case RDRecording::Waiting:
QMessageBox::information(this,tr("Event Active"),
tr("You cannot edit an active event!"));
return;
default:
break;
}
EnableScroll(false);
int id=item->text(28).toInt();
old_conn=GetConnection(item->text(24));
if(old_conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
switch((RDRecording::Type)item->text(29).toInt()) {
case RDRecording::Recording:
recording=new EditRecording(id,&new_events,&catch_filter,this);
if(recording->exec()>=0) {
RefreshLine(item);
new_conn=GetConnection(item->text(24));
if(new_conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
catch_connect[old_conn].connect->removeEvent(id);
catch_connect[new_conn].connect->addEvent(id);
nextEventData();
}
delete recording;
break;
case RDRecording::Playout:
playout=new EditPlayout(id,&new_events,&catch_filter,this,"playout");
if(playout->exec()>=0) {
RefreshLine(item);
new_conn=GetConnection(item->text(24));
if(new_conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
catch_connect[old_conn].connect->removeEvent(id);
catch_connect[new_conn].connect->addEvent(id);
nextEventData();
}
delete playout;
break;
case RDRecording::MacroEvent:
event=new EditCartEvent(id,&new_events,this,"recording");
if(event->exec()>=0) {
RefreshLine(item);
new_conn=GetConnection(item->text(24));
if(new_conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
catch_connect[old_conn].connect->removeEvent(id);
catch_connect[new_conn].connect->addEvent(id);
nextEventData();
}
delete event;
break;
case RDRecording::SwitchEvent:
switch_event=new EditSwitchEvent(id,&new_events,this,"recording");
if(switch_event->exec()>=0) {
RefreshLine(item);
new_conn=GetConnection(item->text(24));
if(new_conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
catch_connect[old_conn].connect->removeEvent(id);
catch_connect[new_conn].connect->addEvent(id);
nextEventData();
}
delete switch_event;
break;
case RDRecording::Download:
download=new EditDownload(id,&new_events,&catch_filter,this,"playout");
if(download->exec()>=0) {
RefreshLine(item);
new_conn=GetConnection(item->text(24));
if(new_conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
catch_connect[old_conn].connect->removeEvent(id);
catch_connect[new_conn].connect->addEvent(id);
nextEventData();
}
delete download;
break;
case RDRecording::Upload:
upload=new EditUpload(id,&new_events,&catch_filter,this,"playout");
if(upload->exec()>=0) {
RefreshLine(item);
new_conn=GetConnection(item->text(24));
if(new_conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
catch_connect[old_conn].connect->removeEvent(id);
catch_connect[new_conn].connect->addEvent(id);
nextEventData();
}
delete upload;
break;
}
ProcessNewRecords(&new_events);
}
void MainWidget::deleteData()
{
QString str;
QString warning;
QString filename;
QString sql;
RDSqlQuery *q;
RDListViewItem *item=(RDListViewItem *)catch_recordings_list->selectedItem();
int conn;
if(!catch_user->editCatches()||(item==NULL)) {
return;
}
EnableScroll(false);
if(item->text(1).isEmpty()) {
str=QString(tr("Are you sure you want to delete event"));
warning=QString().
sprintf("%s\n at %s?",(const char *)str,(const char *)item->text(3));
}
else {
str=QString(tr("Are you sure you want to delete event"));
warning=QString().
sprintf("Are you sure you want to delete event\n\"%s\" at %s?",
(const char *)item->text(1),(const char *)item->text(3));
}
if(QMessageBox::warning(this,tr("Delete Event"),warning,
QMessageBox::Yes,QMessageBox::No)!=
QMessageBox::Yes) {
return;
}
conn=GetConnection(item->text(24));
if(conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
catch_connect[conn].connect->removeEvent(item->text(28).toInt());
sql=QString().sprintf("delete from RECORDINGS where ID=%s",
(const char *)item->text(28));
q=new RDSqlQuery(sql);
delete q;
RDListViewItem *next=(RDListViewItem *)item->nextSibling();
catch_recordings_list->removeItem(item);
if(next!=NULL) {
catch_recordings_list->setSelected(next,true);
}
nextEventData();
}
void MainWidget::ripcConnectedData(bool state)
{
if(!state) {
QMessageBox::warning(this,"Can't Connect","Unable to connect to ripcd!");
exit(0);
}
}
void MainWidget::ripcUserData()
{
QString str;
str=QString("RDCatch")+" v"+VERSION+" - "+tr("Host")+":";
setCaption(str+" "+catch_config->stationName()+", "+tr("User")+": "+
catch_ripc->user());
if(catch_user!=NULL) {
delete catch_user;
}
catch_user=new RDUser(catch_ripc->user());
//
// Set Control Perms
//
bool modification_allowed=catch_user->editCatches();
catch_add_button->setEnabled(modification_allowed);
catch_edit_button->setEnabled(modification_allowed);
catch_delete_button->setEnabled(modification_allowed);
}
void MainWidget::statusChangedData(int serial,unsigned chan,
RDDeck::Status status,int id,
const QString &cutname)
{
// printf("statusChangedData(%d,%u,%d,%d)\n",serial,chan,status,id);
int mon=GetMonitor(serial,chan);
if(id>0) {
RDListViewItem *item=GetItem(id);
if(item!=NULL) {
switch(status) {
case RDDeck::Offline:
item->setBackgroundColor(EVENT_ERROR_COLOR);
break;
case RDDeck::Idle:
item->setBackgroundColor(catch_recordings_list->palette().
color(QPalette::Active,QColorGroup::Base));
break;
case RDDeck::Ready:
item->setBackgroundColor(EVENT_READY_COLOR);
break;
case RDDeck::Waiting:
item->setBackgroundColor(EVENT_WAITING_COLOR);
break;
case RDDeck::Recording:
item->setBackgroundColor(EVENT_ACTIVE_COLOR);
break;
}
item->setText(32,QString().sprintf("%u",status));
UpdateExitCode(item);
}
else {
if(id<RDCATCHD_DYNAMIC_BASE_ID) {
fprintf(stderr,
"rdcatch: received status update for nonexistent ID %d\n",id);
return;
}
}
}
if(mon>=0) {
int waiting_count=0;
int active_count=0;
int waiting_id=0;
RDListViewItem *item=
(RDListViewItem *)catch_recordings_list->firstChild();
while(item!=NULL) {
if(item->text(25).toUInt()==chan) {
switch((RDDeck::Status)item->text(32).toUInt()) {
case RDDeck::Waiting:
active_count++;
waiting_count++;
waiting_id=item->text(28).toInt();
break;
case RDDeck::Ready:
case RDDeck::Recording:
active_count++;
break;
default:
break;
}
}
item=(RDListViewItem *)item->nextSibling();
}
if(waiting_count>1) {
catch_monitor[mon]->deckMon()->setStatus(status,-1,cutname);
}
else {
if((active_count==0)||(status!=RDDeck::Idle)) {
catch_monitor[mon]->deckMon()->setStatus(status,id,cutname);
}
else {
catch_monitor[mon]->deckMon()->
setStatus(RDDeck::Waiting,waiting_id,cutname);
}
}
}
nextEventData();
}
void MainWidget::monitorChangedData(int serial,unsigned chan,bool state)
{
// printf("monitorChangedData(%d,%u,%d)\n",serial,chan,state);
int mon=GetMonitor(serial,chan);
if(mon>=0) {
catch_monitor[mon]->deckMon()->setMonitor(state);
}
}
void MainWidget::scrollButtonData()
{
EnableScroll(!catch_scroll);
}
void MainWidget::reportsButtonData()
{
ListReports *lr=new ListReports(catch_show_today_box->isChecked(),
catch_show_active_box->isChecked(),
catch_dow_box->currentItem(),this);
lr->exec();
delete lr;
}
void MainWidget::headButtonData()
{
RDListViewItem *item=(RDListViewItem *)catch_recordings_list->selectedItem();
if(item==NULL) {
return;
}
EnableScroll(false);
if((!head_playing)&&(!tail_playing)) { // Start Head Play
RDCut *cut=new RDCut(item->text(26));
catch_cae->loadPlay(catch_audition_card,item->text(26),
&catch_audition_stream,&catch_play_handle);
if(catch_audition_stream<0) {
return;
}
RDSetMixerOutputPort(catch_cae,catch_audition_card,catch_audition_stream,
catch_audition_port);
catch_cae->positionPlay(catch_play_handle,cut->startPoint());
catch_cae->setPlayPortActive(catch_audition_card,catch_audition_port,catch_audition_stream);
catch_cae->setOutputVolume(catch_audition_card,catch_audition_stream,catch_audition_port,
0+cut->playGain());
catch_cae->play(catch_play_handle,RDCATCH_AUDITION_LENGTH,
RD_TIMESCALE_DIVISOR,false);
head_playing=true;
delete cut;
}
}
void MainWidget::tailButtonData()
{
RDListViewItem *item=(RDListViewItem *)catch_recordings_list->selectedItem();
if(item==NULL) {
return;
}
EnableScroll(false);
if((!head_playing)&&(!tail_playing)) { // Start Tail Play
RDCut *cut=new RDCut(item->text(26));
catch_cae->loadPlay(catch_audition_card,item->text(26),
&catch_audition_stream,&catch_play_handle);
if(catch_audition_stream<0) {
return;
}
RDSetMixerOutputPort(catch_cae,catch_audition_card,catch_audition_stream,
catch_audition_port);
if((cut->endPoint()-cut->startPoint()-RDCATCH_AUDITION_LENGTH)>0) {
catch_cae->positionPlay(catch_play_handle,
cut->endPoint()-RDCATCH_AUDITION_LENGTH);
}
else {
catch_cae->positionPlay(catch_play_handle,cut->startPoint());
}
catch_cae->setPlayPortActive(catch_audition_card,catch_audition_port,catch_audition_stream);
catch_cae->setOutputVolume(catch_audition_card,catch_audition_stream,catch_audition_port,
0+cut->playGain());
catch_cae->play(catch_play_handle,RDCATCH_AUDITION_LENGTH,
RD_TIMESCALE_DIVISOR,false);
tail_playing=true;
delete cut;
}
}
void MainWidget::stopButtonData()
{
if(head_playing||tail_playing) { // Stop Play
catch_cae->stopPlay(catch_play_handle);
catch_cae->unloadPlay(catch_play_handle);
}
}
void MainWidget::initData(bool state)
{
if(!state) {
QMessageBox::warning(this,tr("Can't Connect"),
tr("Unable to connect to Core AudioEngine"));
exit(1);
}
}
void MainWidget::playedData(int handle)
{
if(head_playing) {
catch_head_button->on();
}
if(tail_playing) {
catch_tail_button->on();
}
catch_stop_button->off();
}
void MainWidget::playStoppedData(int handle)
{
head_playing=false;
tail_playing=false;
catch_head_button->off();
catch_tail_button->off();
catch_stop_button->on();
catch_cae->unloadPlay(catch_play_handle);
}
void MainWidget::meterLevelData(int serial,int deck,int l_r,int level)
{
DeckMon *monitor;
for(unsigned i=0;i<catch_connect[serial].chan.size();i++) {
if(catch_connect[serial].chan[i]==(unsigned)deck) {
monitor=catch_monitor[catch_connect[serial].mon_id[i]]->deckMon();
if(l_r==0) {
monitor->setLeftMeter(level);
}
if(l_r==1) {
monitor->setRightMeter(level);
}
return;
}
}
}
void MainWidget::abortData(int id)
{
catch_connect[catch_monitor[id]->serialNumber()].connect->
stop(catch_monitor[id]->channelNumber());
}
void MainWidget::monitorData(int id)
{
catch_connect[catch_monitor[id]->serialNumber()].connect->
toggleMonitor(catch_monitor[id]->channelNumber());
}
void MainWidget::selectionChangedData(QListViewItem *item)
{
if(item==NULL) {
catch_head_button->setDisabled(true);
catch_tail_button->setDisabled(true);
catch_stop_button->setDisabled(true);
catch_edit_button->setDisabled(true);
return;
}
if(((item->text(29).toInt()==RDRecording::Recording)||
(item->text(29).toInt()==RDRecording::Playout)||
(item->text(29).toInt()==RDRecording::Upload)||
(item->text(29).toInt()==RDRecording::Download))) {
catch_head_button->
setEnabled((catch_audition_card>=0)&&(catch_audition_port>=0));
catch_tail_button->
setEnabled((catch_audition_card>=0)&&(catch_audition_port>=0));
catch_stop_button->setEnabled(true);
}
else {
catch_head_button->setDisabled(true);
catch_tail_button->setDisabled(true);
catch_stop_button->setDisabled(true);
}
}
void MainWidget::doubleClickedData(QListViewItem *,const QPoint &,int)
{
editData();
}
void MainWidget::eventUpdatedData(int id)
{
// printf("eventUpdatedData(%d)\n",id);
RDListViewItem *item=GetItem(id);
if(item==NULL) { // New Event
item=new RDListViewItem(catch_recordings_list);
item->setText(28,QString().sprintf("%d",id));
}
RefreshLine(item);
nextEventData();
}
void MainWidget::eventPurgedData(int id)
{
RDListViewItem *item=GetItem(id);
if(item==NULL) {
return;
}
catch_recordings_list->removeItem(item);
}
void MainWidget::heartbeatFailedData(int id)
{
if(!catch_host_warnings) {
return;
}
QString str;
str=QString(tr("Control connection timed out to host"));
QString msg=QString().sprintf("%s '%s'!",(const char *)str,
(const char *)catch_connect[id].station);
QMessageBox::warning(this,"RDCatch",msg);
}
void MainWidget::quitMainWidget()
{
catch_db->removeDatabase(catch_config->mysqlDbname());
SaveGeometry();
exit(0);
}
void MainWidget::filterChangedData(bool)
{
RDListViewItem *item=(RDListViewItem *)catch_recordings_list->firstChild();
int day=QDate::currentDate().dayOfWeek();
int day_column=0;
switch(day) {
case 1:
case 2:
case 3:
case 4:
case 5:
case 6:
day_column=day+7;
break;
case 7:
day_column=7;
}
if(catch_show_active_box->isChecked()) {
if(catch_show_today_box->isChecked()) {
while(item!=NULL) {
if((item->textColor(1)==EVENT_ACTIVE_TEXT_COLOR)&&
(!item->text(day_column).isEmpty())) {
ShowEvent(item);
}
else {
item->setVisible(false);
}
item=(RDListViewItem *)item->nextSibling();
}
}
else {
if(catch_show_active_box->isChecked()) {
while(item!=NULL) {
if(item->textColor(1)==EVENT_ACTIVE_TEXT_COLOR) {
ShowEvent(item);
}
else {
item->setVisible(false);
}
item=(RDListViewItem *)item->nextSibling();
}
}
}
}
else {
if(catch_show_today_box->isChecked()) {
while(item!=NULL) {
if(!item->text(day_column).isEmpty()) {
ShowEvent(item);
}
else {
item->setVisible(false);
}
item=(RDListViewItem *)item->nextSibling();
}
}
else {
while(item!=NULL) {
ShowEvent(item);
item=(RDListViewItem *)item->nextSibling();
}
}
}
}
void MainWidget::dowChangedData(int id)
{
filterChangedData(false);
}
void MainWidget::clockData()
{
QTime current_time=QTime::currentTime().addMSecs(catch_time_offset);
catch_clock_label->setText(current_time.toString("hh:mm:ss"));
catch_clock_timer->start(1000-current_time.msec(),true);
}
void MainWidget::midnightData()
{
filterChangedData(false);
catch_midnight_timer->
start(86400000+QTime::currentTime().addMSecs(catch_time_offset).
msecsTo(QTime()),true);
}
void MainWidget::closeEvent(QCloseEvent *e)
{
quitMainWidget();
}
void MainWidget::resizeEvent(QResizeEvent *e)
{
assert (e);
assert (catch_monitor_view);
if(catch_monitor.size()<=RDCATCH_MAX_VISIBLE_MONITORS) {
catch_monitor_view->
setGeometry(10,10,e->size().width()-20,32*catch_monitor.size()+4);
catch_monitor_vbox->
setGeometry(0,0,e->size().width()-25,32*catch_monitor.size());
}
else {
catch_monitor_view->
setGeometry(10,10,e->size().width()-20,32*RDCATCH_MAX_VISIBLE_MONITORS);
catch_monitor_vbox->
setGeometry(0,0,e->size().width()-
catch_monitor_view->verticalScrollBar()->geometry().width()-
25,32*catch_monitor.size());
}
int deck_height=0;
if (catch_monitor.size()>0){
deck_height=catch_monitor_view->geometry().y()+
catch_monitor_view->geometry().height();
}
catch_show_active_label->setGeometry(35,deck_height+5,160,20);
catch_show_active_box->setGeometry(15,deck_height+7,15,15);
catch_show_today_label->setGeometry(265,deck_height+5,165,20);
catch_show_today_box->setGeometry(245,deck_height+7,15,15);
catch_dow_label->setGeometry(480,deck_height+5,125,20);
catch_dow_box->setGeometry(610,deck_height+4,120,20);
catch_recordings_list->
setGeometry(10,deck_height+25,e->size().width()-20,
e->size().height()-90-deck_height);
catch_add_button->setGeometry(10,e->size().height()-55,80,50);
catch_edit_button->setGeometry(100,e->size().height()-55,80,50);
catch_delete_button->setGeometry(190,e->size().height()-55,80,50);
catch_scroll_button->setGeometry(290,e->size().height()-55,80,50);
catch_reports_button->setGeometry(380,e->size().height()-55,80,50);
catch_clock_label->setGeometry(470,e->size().height()-38,
e->size().width()-850,20);
catch_head_button->
setGeometry(e->size().width()-370,e->size().height()-55,80,50);
catch_tail_button->
setGeometry(e->size().width()-280,e->size().height()-55,80,50);
catch_stop_button->
setGeometry(e->size().width()-190,e->size().height()-55,80,50);
catch_close_button->
setGeometry(e->size().width()-90,e->size().height()-55,80,50);
}
void MainWidget::ShowEvent(RDListViewItem *item)
{
switch(catch_dow_box->currentItem()) {
case 0: // All Days
item->setVisible(true);
break;
case 1: // Weekdays
if(item->text(8).isEmpty()&&
item->text(9).isEmpty()&&
item->text(10).isEmpty()&&
item->text(11).isEmpty()&&
item->text(12).isEmpty()) {
item->setVisible(false);
}
else {
item->setVisible(true);
}
break;
case 2: // Sunday
if(item->text(7).isEmpty()) {
item->setVisible(false);
}
else {
item->setVisible(true);
}
break;
case 3: // Monday
if(item->text(8).isEmpty()) {
item->setVisible(false);
}
else {
item->setVisible(true);
}
break;
case 4: // Tuesday
if(item->text(9).isEmpty()) {
item->setVisible(false);
}
else {
item->setVisible(true);
}
break;
case 5: // Wednesday
if(item->text(10).isEmpty()) {
item->setVisible(false);
}
else {
item->setVisible(true);
}
break;
case 6: // Thursday
if(item->text(11).isEmpty()) {
item->setVisible(false);
}
else {
item->setVisible(true);
}
break;
case 7: // Friday
if(item->text(12).isEmpty()) {
item->setVisible(false);
}
else {
item->setVisible(true);
}
break;
case 8: // Saturday
if(item->text(13).isEmpty()) {
item->setVisible(false);
}
else {
item->setVisible(true);
}
break;
}
}
int MainWidget::ShowNextEvents(int day,QTime time,QTime *next)
{
RDListViewItem *item=NULL;
QString sql;
int count=0;
if(time.isNull()) {
sql=QString().sprintf("select ID,START_TIME from RECORDINGS \
where (IS_ACTIVE=\"Y\")&& \
(%s=\"Y\") \
order by START_TIME",
(const char *)RDGetShortDayNameEN(day).upper());
}
else {
sql=QString().sprintf("select ID,START_TIME from RECORDINGS \
where (IS_ACTIVE=\"Y\")&& \
(time_to_sec(START_TIME)>time_to_sec(\"%s\"))&& \
(%s=\"Y\") \
order by START_TIME",
(const char *)time.toString("hh:mm:ss"),
(const char *)RDGetShortDayNameEN(day).upper());
}
RDSqlQuery *q=new RDSqlQuery(sql);
if(!q->first()) {
delete q;
return count;
}
*next=q->value(1).toTime();
if((item=GetItem(q->value(0).toInt()))!=NULL) {
if((item->backgroundColor()!=EVENT_ACTIVE_COLOR)&&
(item->backgroundColor()!=EVENT_WAITING_COLOR)){
item->setBackgroundColor(QColor(EVENT_NEXT_COLOR));
}
count++;
}
while(q->next()&&(q->value(1).toTime()==*next)) {
if((item=GetItem(q->value(0).toInt()))!=NULL) {
if((item->backgroundColor()!=EVENT_ACTIVE_COLOR)&&
(item->backgroundColor()!=EVENT_WAITING_COLOR)){
item->setBackgroundColor(QColor(EVENT_NEXT_COLOR));
}
count++;
}
}
delete q;
return count;
}
int MainWidget::AddRecord()
{
QString sql;
RDSqlQuery *q;
int n;
sql=QString("select ID from RECORDINGS order by ID desc limit 1");
q=new RDSqlQuery(sql);
if(q->first()) {
n=q->value(0).toInt()+1;
}
else {
n=1;
}
delete q;
sql=QString().sprintf("insert into RECORDINGS set ID=%d",n);
q=new RDSqlQuery(sql);
delete q;
return n;
}
void MainWidget::ProcessNewRecords(std::vector<int> *adds)
{
int conn=0;
RDListViewItem *item;
for(unsigned i=0;i<adds->size();i++) {
item=new RDListViewItem(catch_recordings_list);
item->setBackgroundColor(catch_recordings_list->palette().color(QPalette::Active,QColorGroup::Base));
item->setText(28,QString().sprintf("%d",adds->at(i)));
RefreshLine(item);
conn=GetConnection(item->text(24));
if(conn<0) {
fprintf(stderr,"rdcatch: invalid connection index!\n");
return;
}
catch_connect[conn].connect->addEvent(adds->at(i));
}
nextEventData();
}
void MainWidget::EnableScroll(bool state)
{
if(state) {
catch_scroll_button->setPalette(catch_scroll_color[0]);
catch_scroll=true;
UpdateScroll();
}
else {
catch_scroll_button->setPalette(catch_scroll_color[1]);
catch_scroll=false;
}
}
void MainWidget::UpdateScroll()
{
RDListViewItem *item=(RDListViewItem *)catch_recordings_list->firstChild();
//
// Search for active event
//
while(item!=NULL) {
if(item->backgroundColor()==EVENT_ACTIVE_COLOR) {
catch_recordings_list->
ensureVisible(0,catch_recordings_list->itemPos(item),
0,catch_recordings_list->size().height()/2);
catch_recordings_list->setCurrentItem(item);
catch_recordings_list->clearSelection();
return;
}
item=(RDListViewItem *)item->nextSibling();
}
//
// Search for next event
//
item=(RDListViewItem *)catch_recordings_list->firstChild();
while(item!=NULL) {
if(item->backgroundColor()==EVENT_NEXT_COLOR) {
catch_recordings_list->
ensureVisible(0,catch_recordings_list->itemPos(item),
0,catch_recordings_list->size().height()/2);
catch_recordings_list->setCurrentItem(item);
catch_recordings_list->clearSelection();
return;
}
item=(RDListViewItem *)item->nextSibling();
}
}
void MainWidget::RefreshList()
{
QString sql;
RDSqlQuery *q;
RDSqlQuery *q1;
RDListViewItem *l;
RDCut *cut;
QString str;
catch_recordings_list->clear();
sql=QString("select RECORDINGS.DESCRIPTION,RECORDINGS.IS_ACTIVE,\
RECORDINGS.STATION_NAME,RECORDINGS.START_TIME,\
RECORDINGS.LENGTH,RECORDINGS.CUT_NAME,RECORDINGS.SUN,\
RECORDINGS.MON,RECORDINGS.TUE,RECORDINGS.WED,RECORDINGS.THU,\
RECORDINGS.FRI,RECORDINGS.SAT,RECORDINGS.SWITCH_INPUT,\
RECORDINGS.START_GPI,RECORDINGS.END_GPI,\
RECORDINGS.TRIM_THRESHOLD,RECORDINGS.STARTDATE_OFFSET,\
RECORDINGS.ENDDATE_OFFSET,RECORDINGS.FORMAT,\
RECORDINGS.CHANNELS,RECORDINGS.SAMPRATE,RECORDINGS.BITRATE,\
RECORDINGS.CHANNEL,RECORDINGS.MACRO_CART,RECORDINGS.ID,\
RECORDINGS.TYPE,RECORDINGS.SWITCH_OUTPUT,RECORDINGS.EXIT_CODE,\
RECORDINGS.ONE_SHOT,RECORDINGS.START_TYPE,\
RECORDINGS.START_LENGTH,RECORDINGS.START_MATRIX,\
RECORDINGS.START_LINE,RECORDINGS.START_OFFSET,\
RECORDINGS.END_TYPE,RECORDINGS.END_TIME,RECORDINGS.END_LENGTH,\
RECORDINGS.END_MATRIX,RECORDINGS.END_LINE,CUTS.ORIGIN_NAME,\
CUTS.ORIGIN_DATETIME,RECORDINGS.URL,RECORDINGS.QUALITY,\
FEEDS.KEY_NAME,EXIT_TEXT from RECORDINGS left join CUTS\
on (RECORDINGS.CUT_NAME=CUTS.CUT_NAME) left join FEEDS \
on (RECORDINGS.FEED_ID=FEEDS.ID)");
// Field Offsets:
//
// 0 - REC.DESCRIPTION 18 - REC.ENDDATE_OFFSET 36 - REC.END_TIME
// 1 - REC.IS_ACTIVE 19 - REC.FORMAT 37 - REC.END_LENGTH
// 2 - REC.STATION_NAME 20 - REC.CHANNELS 38 - REC.END_MATRIX
// 3 - REC.START_TIME 21 - REC.SAMPRATE 39 - REC.END_LINE
// 4 - REC.LENGTH 22 - REC.BITRATE 40 - CUTS.ORIGIN_NAME
// 5 - REC.CUT_NAME 23 - REC.CHANNEL 41 - CUTS.ORIGIN_DATETIME
// 6 - REC.SUN 24 - REC.MACRO_CART 42 - REC.URL
// 7 - REC.MON 25 - REC.ID 43 - REC.QUALITY
// 8 - REC.TUE 26 - REC.TYPE 44 - FEEDS.KEY_NAME
// 9 - REC.WED 27 - REC.SWITCH_OUTPUT 45 - REC.EXIT_TEXT
// 10 - REC.THU 28 - REC.EXIT_CODE
// 11 - REC.FRI 29 - REC.ONE_SHOT
// 12 - REC.SAT 30 - REC.START_TYPE
// 13 - REC.SWITCH_INPUT 31 - REC.START_LENGTH
// 14 - REC.START_GPI 32 - REC.START_MATRIX
// 15 - REC.END_GPI 33 - REC.START_LINE
// 16 - REC.TRIM_THRESHOLD 34 - REC.START_OFFSET
// 17 - REC.STARTDATE_OFFSET 35 - REC.END_TYPE
q=new RDSqlQuery(sql);
while(q->next()) {
l=new RDListViewItem(catch_recordings_list);
l->setBackgroundColor(catch_recordings_list->palette().color(QPalette::Active,QColorGroup::Base));
if(RDBool(q->value(1).toString())) {
l->setTextColor(QColor(EVENT_ACTIVE_TEXT_COLOR));
}
else {
l->setTextColor(QColor(EVENT_INACTIVE_TEXT_COLOR));
}
l->setText(1,q->value(0).toString()); // Description
if(RDBool(q->value(6).toString())) { // Sun
l->setText(7,tr("Su"));
}
if(RDBool(q->value(7).toString())) { // Mon
l->setText(8,tr("Mo"));
}
if(RDBool(q->value(8).toString())) { // Tue
l->setText(9,tr("Tu"));
}
if(RDBool(q->value(9).toString())) { // Wed
l->setText(10,tr("We"));
}
if(RDBool(q->value(10).toString())) { // Thu
l->setText(11,tr("Th"));
}
if(RDBool(q->value(11).toString())) { // Fri
l->setText(12,tr("Fr"));
}
if(RDBool(q->value(12).toString())) { // Sat
l->setText(13,tr("Sa"));
}
switch((RDRecording::Type)q->value(26).toInt()) {
case RDRecording::Recording:
case RDRecording::Playout:
case RDRecording::Download:
case RDRecording::Upload:
l->setText(15,q->value(40).toString()+" - "+
q->value(41).toDateTime().
toString("M/dd/yyyy hh:mm:ss"));
break;
default:
l->setText(15,"");
break;
}
l->setText(16,q->value(29).toString()); // One Shot
str=QString(tr("dB"));
l->setText(17,QString().sprintf("%d %s", // Trim Threshold
-q->value(17).toInt(),(const char *)str));
l->setText(18,q->value(17).toString()); // Startdate Offset
l->setText(19,q->value(18).toString()); // Enddate Offset
l->setText(24,q->value(2).toString()); // Station
l->setText(25,q->value(23).toString()); // Deck
l->setText(26,q->value(5).toString()); // Cut Name
if(q->value(24).toInt()>=0) {
l->setText(27,q->value(24).toString()); // Macro Cart
}
else {
l->setText(27,"");
}
l->setText(28,q->value(25).toString()); // Id
l->setText(29,q->value(26).toString()); // Type
l->setText(32,QString().sprintf("%u",RDDeck::Idle));
switch((RDRecording::Type)q->value(26).toInt()) {
case RDRecording::Recording:
l->setPixmap(0,*catch_record_map);
l->setText(2,QString().sprintf("%s : %dR",
(const char *)q->value(2).toString(),
q->value(23).toInt()));
switch((RDRecording::StartType)q->value(30).toUInt()) {
case RDRecording::HardStart:
str=QString(tr("Hard"));
l->setText(3,q->value(3).toTime().
toString(QString().sprintf("%s: hh:mm:ss",
(const char *)str)));
break;
case RDRecording::GpiStart:
str=QString(tr("Gpi"));
l->setText(3,QString().
sprintf("%s: %s,%s,%d:%d,%s",
(const char *)str,
(const char *)q->value(3).
toTime().toString("hh:mm:ss"),
(const char *)q->value(3).toTime().
addMSecs(q->value(31).toInt()).
toString("hh:mm:ss"),
q->value(32).toInt(),
q->value(33).toInt(),
(const char *)QTime().
addMSecs(q->value(34).toUInt()).
toString("mm:ss")));
break;
}
switch((RDRecording::EndType)q->value(35).toUInt()) {
case RDRecording::LengthEnd:
str=QString(tr("Len"));
l->setText(4,QString().sprintf("%s: %s",
(const char *)str,
(const char *)
RDGetTimeLength(q->value(4).toUInt(),
false,false)));
break;
case RDRecording::HardEnd:
str=QString(tr("Hard"));
l->setText(4,QString().sprintf("%s: %s",
(const char *)str,
(const char *)
q->value(36).toTime().
toString("hh:mm:ss")));
break;
case RDRecording::GpiEnd:
str=QString(tr("Gpi"));
l->setText(4,QString().
sprintf("%s: %s,%s,%d:%d",
(const char *)str,
(const char *)q->value(36).
toTime().toString("hh:mm:ss"),
(const char *)q->value(36).toTime().
addMSecs(q->value(37).toInt()).
toString("hh:mm:ss"),
q->value(38).toInt(),
q->value(39).toInt()));
break;
}
str=QString(tr("Cut"));
l->setText(6,QString().
sprintf("%s %s",(const char *)str,
(const char *)q->value(5).toString()));
sql=QString().sprintf("select SWITCH_STATION,SWITCH_MATRIX\
from DECKS where \
(STATION_NAME=\"%s\")&&(CHANNEL=%d)",
(const char *)q->value(2).toString(),
q->value(23).toInt());
q1=new RDSqlQuery(sql);
if(q1->first()) { // Source
l->setText(5,GetSourceName(q1->value(0).toString(),
q1->value(1).toInt(),
q->value(13).toInt()));
}
delete q1;
switch((RDSettings::Format)q->value(19).toInt()) { // Format
case RDSettings::Pcm16:
l->setText(20,tr("PCM16"));
break;
case RDSettings::MpegL1:
l->setText(20,tr("MPEG Layer 1"));
break;
case RDSettings::MpegL2:
case RDSettings::MpegL2Wav:
l->setText(20,tr("MPEG Layer 2"));
break;
case RDSettings::MpegL3:
l->setText(20,tr("MPEG Layer 3"));
break;
case RDSettings::Flac:
l->setText(20,tr("FLAC"));
break;
case RDSettings::OggVorbis:
l->setText(20,tr("OggVorbis"));
break;
}
l->setText(21,q->value(20).toString()); // Channels
l->setText(22,q->value(21).toString()); // Sample Rate
l->setText(23,q->value(22).toString()); // Bit Rate
break;
case RDRecording::Playout:
l->setPixmap(0,*catch_playout_map);
l->setText(2,QString().sprintf("%s : %dP",
(const char *)q->value(2).toString(),
q->value(23).toInt()-128));
l->setText(3,q->value(3).toTime().toString("Hard: hh:mm:ss"));
cut=new RDCut(q->value(5).toString());
str=QString(tr("Len"));
if(cut->exists()) {
l->setText(4,QString().sprintf("%s: %s",(const char *)str,
(const char *)RDGetTimeLength(cut->length(),false,false)));
}
delete cut;
str=QString(tr("Cut"));
l->setText(5,QString().
sprintf("%s %s",(const char *)str,
(const char *)q->value(5).toString()));
break;
case RDRecording::MacroEvent:
l->setPixmap(0,*catch_macro_map);
l->setText(2,q->value(2).toString());
str=QString(tr("Hard"));
l->setText(3,q->value(3).toTime().
toString(QString().
sprintf("%s: hh:mm:ss",(const char *)str)));
str=QString(tr("Cart"));
l->setText(5,QString().sprintf("%s %06d",(const char *)str,
q->value(24).toInt()));
break;
case RDRecording::SwitchEvent:
l->setPixmap(0,*catch_switch_map);
l->setText(2,q->value(2).toString());
str=QString(tr("Hard"));
l->setText(3,q->value(3).toTime().
toString(QString().sprintf("%s: hh:mm:ss",(const char *)str)));
l->setText(5,GetSourceName(q->value(2).toString(), // Source
q->value(23).toInt(),
q->value(13).toInt()));
l->setText(6,GetDestinationName(q->value(2).toString(), // Dest
q->value(23).toInt(),
q->value(27).toInt()));
break;
case RDRecording::Download:
l->setPixmap(0,*catch_download_map);
l->setText(2,q->value(2).toString());
str=QString(tr("Hard"));
l->setText(3,q->value(3).toTime().
toString(QString().
sprintf("%s: hh:mm:ss",(const char *)str)));
str=QString(tr("Cut"));
l->setText(5,q->value(42).toString());
l->setText(6,QString().
sprintf("%s %s",(const char *)str,
(const char *)q->value(5).toString()));
break;
case RDRecording::Upload:
l->setPixmap(0,*catch_upload_map);
l->setText(2,q->value(2).toString());
str=QString(tr("Hard"));
l->setText(3,q->value(3).toTime().
toString(QString().
sprintf("%s: hh:mm:ss",(const char *)str)));
str=QString(tr("Cut"));
l->setText(5,QString().
sprintf("%s %s",(const char *)str,
(const char *)q->value(5).toString()));
l->setText(6,q->value(42).toString());
switch((RDSettings::Format)q->value(19).toInt()) { // Format
case RDSettings::Pcm16:
l->setText(20,tr("PCM16"));
break;
case RDSettings::MpegL1:
l->setText(20,tr("MPEG Layer 1"));
break;
case RDSettings::MpegL2:
case RDSettings::MpegL2Wav:
l->setText(20,tr("MPEG Layer 2"));
break;
case RDSettings::MpegL3:
l->setText(20,tr("MPEG Layer 3"));
break;
case RDSettings::Flac:
l->setText(20,tr("FLAC"));
break;
case RDSettings::OggVorbis:
l->setText(20,tr("OggVorbis"));
break;
}
if(q->value(44).toString().isEmpty()) {
l->setText(14,tr("[none]"));
}
else {
l->setText(14,q->value(44).toString()); // Feed Key Name
}
l->setText(21,q->value(20).toString()); // Channels
l->setText(22,q->value(21).toString()); // Sample Rate
if(q->value(22).toInt()==0) { // Bit Rate/Quality
l->setText(23,QString().sprintf("Qual %d",q->value(43).toInt()));
}
else {
l->setText(23,QString().sprintf("%d kb/sec",
q->value(22).toInt()/1000));
}
break;
}
DisplayExitCode(l,(RDRecording::ExitCode)q->value(28).toInt(),
q->value(45).toString());
}
delete q;
}
void MainWidget::RefreshLine(RDListViewItem *item)
{
QString sql;
RDSqlQuery *q;
RDSqlQuery *q1;
RDCut *cut;
QString str;
int id=item->text(28).toInt();
sql=QString().sprintf("select RECORDINGS.DESCRIPTION,\
RECORDINGS.IS_ACTIVE,\
RECORDINGS.STATION_NAME,\
RECORDINGS.START_TIME,\
RECORDINGS.LENGTH,\
RECORDINGS.CUT_NAME,\
RECORDINGS.SUN,\
RECORDINGS.MON,\
RECORDINGS.TUE,\
RECORDINGS.WED,\
RECORDINGS.THU,\
RECORDINGS.FRI,\
RECORDINGS.SAT,\
RECORDINGS.SWITCH_INPUT,\
RECORDINGS.START_GPI,\
RECORDINGS.END_GPI,\
RECORDINGS.TRIM_THRESHOLD,\
RECORDINGS.STARTDATE_OFFSET,\
RECORDINGS.ENDDATE_OFFSET,\
RECORDINGS.FORMAT,\
RECORDINGS.CHANNELS,\
RECORDINGS.SAMPRATE,\
RECORDINGS.BITRATE,\
RECORDINGS.CHANNEL,\
RECORDINGS.MACRO_CART,\
RECORDINGS.TYPE,\
RECORDINGS.SWITCH_OUTPUT,\
RECORDINGS.EXIT_CODE,\
RECORDINGS.ONE_SHOT,\
RECORDINGS.START_TYPE,\
RECORDINGS.START_LENGTH,\
RECORDINGS.START_MATRIX,\
RECORDINGS.START_LINE,\
RECORDINGS.START_OFFSET,\
RECORDINGS.END_TYPE,\
RECORDINGS.END_TIME,\
RECORDINGS.END_LENGTH,\
RECORDINGS.END_MATRIX,\
RECORDINGS.END_LINE,\
CUTS.ORIGIN_NAME,\
CUTS.ORIGIN_DATETIME,\
RECORDINGS.URL,\
RECORDINGS.QUALITY,\
FEEDS.KEY_NAME,\
RECORDINGS.EXIT_TEXT \
from RECORDINGS left join CUTS \
on (RECORDINGS.CUT_NAME=CUTS.CUT_NAME) left join FEEDS\
on (RECORDINGS.FEED_ID=FEEDS.ID) \
where RECORDINGS.ID=%d",id);
q=new RDSqlQuery(sql);
if(q->first()) {
if(RDBool(q->value(1).toString())) {
item->setTextColor(QColor(EVENT_ACTIVE_TEXT_COLOR));
}
else {
item->setTextColor(QColor(EVENT_INACTIVE_TEXT_COLOR));
}
item->setText(1,q->value(0).toString()); // Description
if(q->value(25).toInt()==RDRecording::Recording) {
item->setText(2,q->value(2).toString()+QString(" : ")+ // Station : Deck
q->value(23).toString());
}
else {
item->setText(2,q->value(2).toString());
}
item->setText(4,RDGetTimeLength(q->value(4).toUInt())); // Length
if(RDBool(q->value(6).toString())) { // Sun
item->setText(7,tr("Su"));
}
else {
item->setText(7,"");
}
if(RDBool(q->value(7).toString())) { // Mon
item->setText(8,tr("Mo"));
}
else {
item->setText(8,"");
}
if(RDBool(q->value(8).toString())) { // Tue
item->setText(9,tr("Tu"));
}
else {
item->setText(9,"");
}
if(RDBool(q->value(9).toString())) { // Wed
item->setText(10,tr("We"));
}
else {
item->setText(10,"");
}
if(RDBool(q->value(10).toString())) { // Thu
item->setText(11,tr("Th"));
}
else {
item->setText(11,"");
}
if(RDBool(q->value(11).toString())) { // Fri
item->setText(12,tr("Fr"));
}
else {
item->setText(12,"");
}
if(RDBool(q->value(12).toString())) { // Sat
item->setText(13,tr("Sa"));
}
else {
item->setText(13,"");
}
switch((RDRecording::Type)q->value(25).toInt()) {
case RDRecording::Recording:
case RDRecording::Playout:
case RDRecording::Download:
case RDRecording::Upload:
item->setText(15,q->value(39).toString()+" - "+
q->value(40).toDateTime().
toString("M/dd/yyyy hh:mm:ss"));
break;
default:
item->setText(15,"");
}
item->setText(16,q->value(28).toString()); // One Shot
str=QString(tr("dB"));
item->setText(17,QString().sprintf("%d %s", // Trim Threshold
-q->value(17).toInt(),(const char *)str));
item->setText(18,q->value(17).toString()); // Startdate Offset
item->setText(19,q->value(18).toString()); // Enddate Offset
/*
*/
item->setText(24,q->value(2).toString()); // Station
item->setText(25,q->value(23).toString()); // Deck
item->setText(26,q->value(5).toString()); // Cut Name
if(q->value(24).toInt()>0) {
item->setText(27,q->value(24).toString()); // Macro Cart
}
else {
item->setText(27,"");
}
item->setText(29,q->value(25).toString()); // Type
switch((RDRecording::Type)q->value(25).toInt()) { // Source
case RDRecording::Recording:
item->setText(2,QString().sprintf("%s : %dR",
(const char *)q->value(2).toString(),
q->value(23).toInt()));
switch((RDRecording::StartType)q->value(29).toUInt()) {
case RDRecording::HardStart:
str=QString(tr("Hard"));
item->
setText(3,q->value(3).toTime().
toString(QString().
sprintf("%s: hh:mm:ss",(const char *)str)));
break;
case RDRecording::GpiStart:
str=QString(tr("Gpi"));
item->setText(3,QString().
sprintf("%s: %s,%s,%d:%d,%s",
(const char *)str,
(const char *)q->value(3).
toTime().toString("hh:mm:ss"),
(const char *)q->value(3).toTime().
addMSecs(q->value(30).toInt()).
toString("hh:mm:ss"),
q->value(31).toInt(),
q->value(32).toInt(),
(const char *)QTime().
addMSecs(q->value(33).toUInt()).
toString("mm:ss")));
break;
}
switch((RDRecording::EndType)q->value(34).toUInt()) {
case RDRecording::LengthEnd:
str=QString(tr("Len"));
item->setText(4,QString().sprintf("%s: %s",(const char *)str,
(const char *)
RDGetTimeLength(q->value(4).toUInt(),
false,false)));
break;
case RDRecording::HardEnd:
str=QString(tr("Hard"));
item->setText(4,QString().sprintf("%s: %s",(const char *)str,
(const char *)
q->value(35).toTime().
toString("hh:mm:ss")));
break;
case RDRecording::GpiEnd:
str=QString(tr("Gpi"));
item->setText(4,QString().
sprintf("%s: %s,%s,%d:%d",(const char *)str,
(const char *)q->value(35).
toTime().toString("hh:mm:ss"),
(const char *)q->value(35).toTime().
addMSecs(q->value(36).toInt()).
toString("hh:mm:ss"),
q->value(37).toInt(),
q->value(38).toInt()));
break;
}
sql=QString().sprintf("select SWITCH_STATION,SWITCH_MATRIX\
from DECKS where \
(STATION_NAME=\"%s\")&&(CHANNEL=%d)",
(const char *)q->value(2).toString(),
q->value(23).toInt());
q1=new RDSqlQuery(sql);
if(q1->first()) {
item->setText(5,GetSourceName(q1->value(0).toString(),
q1->value(1).toInt(),
q->value(13).toInt()));
}
delete q1;
item->setText(6,QString().
sprintf("Cut %s",(const char *)q->value(5).toString()));
item->setPixmap(0,*catch_record_map);
switch((RDSettings::Format)q->value(19).toInt()) { // Format
case RDSettings::Pcm16:
item->setText(20,tr("PCM16"));
break;
case RDSettings::MpegL1:
item->setText(20,tr("MPEG Layer 1"));
break;
case RDSettings::MpegL2:
case RDSettings::MpegL2Wav:
item->setText(20,tr("MPEG Layer 2"));
break;
case RDSettings::MpegL3:
item->setText(20,tr("MPEG Layer 3"));
break;
case RDSettings::Flac:
item->setText(20,tr("FLAC"));
break;
case RDSettings::OggVorbis:
item->setText(20,tr("OggVorbis"));
break;
}
item->setText(21,q->value(20).toString()); // Channels
item->setText(22,q->value(21).toString()); // Sample Rate
item->setText(23,q->value(22).toString()); // Bit Rate
break;
case RDRecording::Playout:
item->setText(2,QString().sprintf("%s : %dP",
(const char *)q->value(2).toString(),
q->value(23).toInt()-128));
item->setText(3,q->value(3).toTime().toString("Hard: hh:mm:ss"));
cut=new RDCut(q->value(5).toString());
if(cut->exists()) {
str=QString(tr("Len"));
item->setText(4,QString().sprintf("%s: %s",(const char *)str,
(const char *)RDGetTimeLength(cut->length(),false,false)));
}
else {
item->setText(4,"");
}
delete cut;
str=QString(tr("Cut"));
item->setText(5,QString().
sprintf("%s %s",(const char *)str,
(const char *)q->value(5).toString()));
item->setPixmap(0,*catch_playout_map);
item->setText(20,"");
item->setText(21,"");
item->setText(22,"");
item->setText(23,"");
break;
case RDRecording::MacroEvent:
str=QString(tr("Hard"));
item->setText(3,q->value(3).toTime().
toString(QString().
sprintf("%s: hh:mm:ss",(const char *)str)));
item->setText(4,"");
item->setText(5,QString().sprintf("Cart %06d",q->value(24).toInt()));
item->setPixmap(0,*catch_macro_map);
item->setText(20,"");
item->setText(21,"");
item->setText(22,"");
item->setText(23,"");
break;
case RDRecording::SwitchEvent:
str=QString(tr("Hard"));
item->setText(3,q->value(3).toTime().
toString(QString().
sprintf("%s: hh:mm:ss",(const char *)str)));
item->setText(4,"");
item->setText(5,GetSourceName(q->value(2).toString(),
q->value(23).toInt(),
q->value(13).toInt()));
item->setPixmap(0,*catch_switch_map);
item->setText(6,GetDestinationName(q->value(2).toString(), // Dest
q->value(23).toInt(),
q->value(26).toInt()));
item->setText(20,"");
item->setText(21,"");
item->setText(22,"");
item->setText(23,"");
break;
case RDRecording::Download:
item->setPixmap(0,*catch_download_map);
str=QString(tr("Hard"));
item->setText(2,q->value(2).toString());
item->setText(3,q->value(3).toTime().
toString(QString().
sprintf("%s: hh:mm:ss",(const char *)str)));
item->setText(4,"");
item->setText(5,q->value(41).toString());
item->setText(6,QString().
sprintf("Cut %s",(const char *)q->value(5).toString()));
item->setText(20,"");
item->setText(21,"");
item->setText(22,"");
item->setText(23,"");
break;
case RDRecording::Upload:
item->setPixmap(0,*catch_upload_map);
str=QString(tr("Hard"));
item->setText(2,q->value(2).toString());
item->setText(3,q->value(3).toTime().
toString(QString().
sprintf("%s: hh:mm:ss",(const char *)str)));
item->setText(4,"");
item->setText(5,QString().
sprintf("Cut %s",(const char *)q->value(5).toString()));
item->setText(6,q->value(41).toString());
if(q->value(43).toString().isEmpty()) {
item->setText(14,tr("[none]"));
}
else {
item->setText(14,q->value(43).toString());
}
switch((RDSettings::Format)q->value(19).toInt()) { // Format
case RDSettings::Pcm16:
item->setText(20,tr("PCM16"));
break;
case RDSettings::MpegL1:
item->setText(20,tr("MPEG Layer 1"));
break;
case RDSettings::MpegL2:
case RDSettings::MpegL2Wav:
item->setText(20,tr("MPEG Layer 2"));
break;
case RDSettings::MpegL3:
item->setText(20,tr("MPEG Layer 3"));
break;
case RDSettings::Flac:
item->setText(20,tr("FLAC"));
break;
case RDSettings::OggVorbis:
item->setText(20,tr("OggVorbis"));
break;
}
item->setText(21,q->value(20).toString()); // Channels
item->setText(22,q->value(21).toString()); // Sample Rate
item->setText(23,q->value(22).toString()); // Bit Rate
if(q->value(22).toInt()==0) { // Bit Rate/Quality
item->setText(23,QString().sprintf("Qual %d",
q->value(42).toInt()));
}
else {
item->setText(23,QString().sprintf("%d kb/sec",
q->value(22).toInt()/1000));
}
break;
}
DisplayExitCode(item,(RDRecording::ExitCode)q->value(27).toInt(),
q->value(44).toString());
}
else { // Event removed
delete item;
}
delete q;
}
void MainWidget::UpdateExitCode(RDListViewItem *item)
{
RDRecording::ExitCode code=RDRecording::InternalError;
QString err_text=tr("Unknown");
QString sql=QString().sprintf("select RECORDINGS.EXIT_CODE,\
CUTS.ORIGIN_NAME,CUTS.ORIGIN_DATETIME,\
RECORDINGS.EXIT_TEXT \
from RECORDINGS left join CUTS \
on RECORDINGS.CUT_NAME=CUTS.CUT_NAME \
where RECORDINGS.ID=%d",
item->text(28).toInt());
RDSqlQuery *q=new RDSqlQuery(sql);
if(q->first()) {
code=(RDRecording::ExitCode)q->value(0).toInt();
err_text=q->value(3).toString();
item->setText(15,q->value(1).toString()+" - "+q->value(2).toDateTime().
toString("M/dd/yyyy hh:mm:ss"));
}
else {
item->setText(15,"");
}
delete q;
DisplayExitCode(item,code,err_text);
}
void MainWidget::DisplayExitCode(RDListViewItem *item,
RDRecording::ExitCode code,
const QString &err_text)
{
item->setText(31,QString().sprintf("%u",code));
switch(code) {
case RDRecording::Ok:
case RDRecording::Downloading:
case RDRecording::Uploading:
case RDRecording::RecordActive:
case RDRecording::PlayActive:
case RDRecording::Waiting:
item->setText(30,RDRecording::exitString(code));
break;
case RDRecording::Short:
case RDRecording::LowLevel:
case RDRecording::HighLevel:
case RDRecording::Interrupted:
case RDRecording::DeviceBusy:
case RDRecording::NoCut:
case RDRecording::UnknownFormat:
item->setText(30,RDRecording::exitString(code));
item->setBackgroundColor(EVENT_ERROR_COLOR);
break;
case RDRecording::ServerError:
case RDRecording::InternalError:
item->setText(30,RDRecording::exitString(code)+": "+err_text);
item->setBackgroundColor(EVENT_ERROR_COLOR);
break;
}
}
QString MainWidget::GetSourceName(QString station,int matrix,int input)
{
QString input_name;
QString sql=QString().sprintf("select NAME from INPUTS where \
(STATION_NAME=\"%s\")&&\
(MATRIX=%d)&&(NUMBER=%d)",
(const char *)station,
matrix,input);
RDSqlQuery *q=new RDSqlQuery(sql);
if(q->first()) {
input_name=q->value(0).toString();
}
delete q;
return input_name;
}
QString MainWidget::GetDestinationName(QString station,int matrix,int output)
{
QString output_name;
QString sql=QString().sprintf("select NAME from OUTPUTS where \
(STATION_NAME=\"%s\")&&\
(MATRIX=%d)&&(NUMBER=%d)",
(const char *)station,
matrix,output);
RDSqlQuery *q=new RDSqlQuery(sql);
if(q->first()) {
output_name=q->value(0).toString();
}
delete q;
return output_name;
}
RDListViewItem *MainWidget::GetItem(int id)
{
RDListViewItem *item=(RDListViewItem *)catch_recordings_list->firstChild();
if(item==NULL) {
return NULL;
}
do {
if(item->text(28).toInt()==id) {
return item;
}
} while ((item=(RDListViewItem *)item->nextSibling())!=NULL);
return NULL;
}
int MainWidget::GetMonitor(int serial,int chan)
{
for(unsigned i=0;i<catch_monitor.size();i++) {
if((catch_monitor[i]->serialNumber()==serial)&&
(catch_monitor[i]->channelNumber()==chan)) {
return i;
}
}
return -1;
}
int MainWidget::GetConnection(QString station,unsigned chan)
{
for(int i=0;i<catch_station_count;i++) {
if(catch_connect[i].station==station.lower()) {
if(chan==0) {
return i;
}
for(unsigned j=0;j<catch_connect[i].chan.size();j++) {
printf("J: %u\n",j);
if(catch_connect[i].chan[j]==chan) {
return i;
}
}
return -1;
}
}
printf(" No connection found!\n");
return -1;
}
QString MainWidget::GeometryFile() {
bool home_found = false;
QString home = RDGetHomeDir(&home_found);
if (home_found) {
return home + "/" + RDCATCH_GEOMETRY_FILE;
} else {
return NULL;
}
}
void MainWidget::LoadGeometry()
{
QString geometry_file = GeometryFile();
if(geometry_file==NULL) {
return;
}
RDProfile *profile=new RDProfile();
profile->setSource(geometry_file);
resize(profile->intValue("RDCatch","Width",sizeHint().width()),
profile->intValue("RDCatch","Height",sizeHint().height()));
delete profile;
}
void MainWidget::SaveGeometry()
{
QString geometry_file = GeometryFile();
if(geometry_file==NULL) {
return;
}
FILE *file=fopen(geometry_file,"w");
if(file==NULL) {
return;
}
fprintf(file,"[RDCatch]\n");
fprintf(file,"Width=%d\n",geometry().width());
fprintf(file,"Height=%d\n",geometry().height());
fclose(file);
}
int main(int argc,char *argv[])
{
QApplication a(argc,argv);
//
// Load Translations
//
QTranslator qt(0);
qt.load(QString(QTDIR)+QString("/translations/qt_")+QTextCodec::locale(),
".");
a.installTranslator(&qt);
QTranslator rd(0);
rd.load(QString(PREFIX)+QString("/share/rivendell/librd_")+
QTextCodec::locale(),".");
a.installTranslator(&rd);
QTranslator rdhpi(0);
rdhpi.load(QString(PREFIX)+QString("/share/rivendell/librdhpi_")+
QTextCodec::locale(),".");
a.installTranslator(&rdhpi);
QTranslator tr(0);
tr.load(QString(PREFIX)+QString("/share/rivendell/rdcatch_")+
QTextCodec::locale(),".");
a.installTranslator(&tr);
//
// Start Event Loop
//
MainWidget *w=new MainWidget(NULL,"main");
a.setMainWidget(w);
w->show();
return a.exec();
}