You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
digikam/digikam/digikam/cameralist.cpp

277 lines
6.8 KiB

/* ============================================================
*
* This file is a part of digiKam project
* http://www.digikam.org
*
* Date : 2003-02-03
* Description : Cameras list container
*
* Copyright (C) 2003-2005 by Renchi Raju <renchi@pooh.tam.uiuc.edu>
* Copyright (C) 2006-2007 by Gilles Caulier <caulier dot gilles at gmail dot com>
*
* 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, 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.
*
* ============================================================ */
// TQt includes.
#include <tqdatetime.h>
#include <tqstring.h>
#include <tqfile.h>
#include <tqdom.h>
#include <tqtextstream.h>
// KDE includes.
#include <tdemessagebox.h>
#include <tdelocale.h>
// Local includes.
#include "gpcamera.h"
#include "cameratype.h"
#include "cameralist.h"
#include "cameralist.moc"
namespace Digikam
{
CameraList* CameraList::m_instance = 0;
CameraList* CameraList::instance()
{
return m_instance;
}
class CameraListPrivate
{
public:
bool modified;
TQPtrList<CameraType> clist;
TQString file;
};
CameraList::CameraList(TQObject *parent, const TQString& file)
: TQObject(parent)
{
d = new CameraListPrivate;
d->clist.setAutoDelete(true);
d->file = file;
d->modified = false;
m_instance = this;
}
CameraList::~CameraList()
{
save();
d->clist.clear();
delete d;
m_instance = 0;
}
bool CameraList::load()
{
d->modified = false;
TQFile cfile(d->file);
if (!cfile.open(IO_ReadOnly))
return false;
TQDomDocument doc("cameralist");
if (!doc.setContent(&cfile))
return false;
TQDomElement docElem = doc.documentElement();
if (docElem.tagName()!="cameralist")
return false;
for (TQDomNode n = docElem.firstChild();
!n.isNull(); n = n.nextSibling())
{
TQDomElement e = n.toElement();
if (e.isNull()) continue;
if (e.tagName() != "item") continue;
TQString title = e.attribute("title");
TQString model = e.attribute("model");
TQString port = e.attribute("port");
TQString path = e.attribute("path");
TQDateTime lastAccess = TQDateTime::currentDateTime();
if (!e.attribute("lastaccess").isEmpty())
lastAccess = TQDateTime::fromString(e.attribute("lastaccess"), TQt::ISODate);
CameraType *ctype = new CameraType(title, model, port, path, lastAccess);
insertPrivate(ctype);
}
return true;
}
bool CameraList::save()
{
// If not modified don't save the file
if (!d->modified)
return true;
TQDomDocument doc("cameralist");
doc.setContent(TQString("<!DOCTYPE XMLCameraList><cameralist version=\"1.1\" client=\"digikam\"/>"));
TQDomElement docElem=doc.documentElement();
for (CameraType *ctype = d->clist.first(); ctype;
ctype = d->clist.next())
{
TQDomElement elem = doc.createElement("item");
elem.setAttribute("title", ctype->title());
elem.setAttribute("model", ctype->model());
elem.setAttribute("port", ctype->port());
elem.setAttribute("path", ctype->path());
elem.setAttribute("lastaccess", ctype->lastAccess().toString(TQt::ISODate));
docElem.appendChild(elem);
}
TQFile cfile(d->file);
if (!cfile.open(IO_WriteOnly))
return false;
TQTextStream stream(&cfile);
stream.setEncoding(TQTextStream::UnicodeUTF8);
stream << doc.toString();
cfile.close();
return true;
}
bool CameraList::changeCameraAccessTime(const TQString& cameraTitle, const TQDateTime& newDate)
{
CameraType* cam = find(cameraTitle);
if (cam)
{
cam->setLastAccess(newDate);
d->modified = true;
save();
return true;
}
return false;
}
void CameraList::insert(CameraType* ctype)
{
if (!ctype) return;
d->modified = true;
insertPrivate(ctype);
}
void CameraList::remove(CameraType* ctype)
{
if (!ctype) return;
d->modified = true;
removePrivate(ctype);
}
void CameraList::insertPrivate(CameraType* ctype)
{
if (!ctype) return;
emit signalCameraAdded(ctype);
d->clist.append(ctype);
}
void CameraList::removePrivate(CameraType* ctype)
{
if (!ctype) return;
emit signalCameraRemoved(ctype);
d->clist.remove(ctype);
}
TQPtrList<CameraType>* CameraList::cameraList()
{
return &d->clist;
}
CameraType* CameraList::find(const TQString& title)
{
for (CameraType *ctype = d->clist.first(); ctype;
ctype = d->clist.next())
{
if (ctype->title() == title)
return ctype;
}
return 0;
}
CameraType* CameraList::autoDetect(bool& retry)
{
retry = false;
TQString model, port;
if (GPCamera::autoDetect(model, port) != 0)
{
retry = ( KMessageBox::warningYesNo(0, i18n("Failed to auto-detect camera; "
"please make sure it is connected "
"properly and is turned on. "
"Would you like to try again?"))
== KMessageBox::Yes );
return 0;
}
// check if the camera is already in the list
for (CameraType *ctype = d->clist.first(); ctype;
ctype = d->clist.next())
{
// we can get away with checking only the model, as the auto-detection
// works only for usb cameras. so the port is always usb:
if (ctype->model() == model)
return ctype;
}
// looks like a new camera
// NOTE: libgphoto2 now (2.1.4+) expects port names to be
// something like "usb:001,012". but on linux these port numbers
// will change every time camera is reconnected. gphoto port funcs
// also allow regexp match, so the safe bet is to just pass in
// "usb:" and cross your fingers that user doesn't have multiple cameras
// connected at the same time (whack them if they do).
if (port.startsWith("usb:"))
port = "usb:";
CameraType* ctype = new CameraType(model, model, port, "/", TQDateTime::currentDateTime());
insert(ctype);
return ctype;
}
void CameraList::clear()
{
CameraType *ctype = d->clist.first();
while (ctype)
{
remove(ctype);
ctype = d->clist.first();
}
}
} // namespace Digikam