playlistsplitter.cpp 19.3 KB
Newer Older
1 2 3 4 5
/***************************************************************************
                          playlistsplitter.cpp  -  description
                             -------------------
    begin                : Fri Sep 13 2002
    copyright            : (C) 2002 by Scott Wheeler
6
    email                : wheeler@kde.org
7 8 9 10 11 12 13 14 15 16 17
 ***************************************************************************/

/***************************************************************************
 *                                                                         *
 *   This program is free software; you can redistribute it and/or modify  *
 *   it under the terms of the GNU General Public License as published by  *
 *   the Free Software Foundation; either version 2 of the License, or     *
 *   (at your option) any later version.                                   *
 *                                                                         *
 ***************************************************************************/

18
#include <klocale.h>
19
#include <kiconloader.h>
20
#include <kapplication.h>
21
#include <kstandarddirs.h>
22
#include <kmessagebox.h>
23
#include <kdirwatch.h>
Nadeem Hasan's avatar
Nadeem Hasan committed
24
#include <klineeditdlg.h>
25 26
#include <kdebug.h>

27
#include <qtimer.h>
28

29
#include "playlistitem.h"
30
#include "playlistsplitter.h"
31
#include "collectionlist.h"
32
#include "directorylist.h"
33
#include "playlist.h"
34
#include "playlistsearch.h"
35

36 37
QStringList *PlaylistSplitter::m_mediaExtensions = 0;
QStringList *PlaylistSplitter::m_listExtensions = 0;
38 39

////////////////////////////////////////////////////////////////////////////////
40
// helper functions
41 42
////////////////////////////////////////////////////////////////////////////////

43
void processEvents()
44
{
45 46 47
    static int processed = 0;
    if(processed == 0)
        kapp->processEvents();
48
    processed = (processed + 1) % 5;
49 50
}

51
////////////////////////////////////////////////////////////////////////////////
52
// public methods
53 54
////////////////////////////////////////////////////////////////////////////////

55 56 57 58
PlaylistSplitter::PlaylistSplitter(QWidget *parent, bool restore, const char *name) : 
    QSplitter(Qt::Horizontal, parent, name),
    m_playingItem(0), m_searchWidget(0), m_dynamicList(0), m_restore(restore), 
    m_nextPlaylistItem(0)
59
{
60 61 62 63 64 65 66 67
    if(!m_mediaExtensions && !m_listExtensions) {
	m_mediaExtensions = new QStringList();
	m_listExtensions = new QStringList();

	m_mediaExtensions->append("mp3");
	m_mediaExtensions->append("ogg");
	m_listExtensions->append("m3u");
    }
68 69 70

    setupLayout();
    readConfig();
71 72

    m_editor->slotUpdateCollection();
73
}
74

75
PlaylistSplitter::~PlaylistSplitter()
76
{
Scott Wheeler's avatar
Scott Wheeler committed
77
    delete m_dirWatch;
78
    saveConfig();
79 80
}

81
QString PlaylistSplitter::uniquePlaylistName(const QString &startingWith, bool useParenthesis)
82
{
83
    if(!m_playlistBox)
84
	return QString::null;
85

86
    QStringList names = m_playlistBox->names();
87 88 89 90 91 92

    int playlistNumber = 1;

    // while the list contains more than zero instances of the generated 
    // string...

93 94 95 96
    if(useParenthesis) {
	while(names.contains(startingWith + " (" + QString::number(playlistNumber) + ")") != 0)
	    playlistNumber++;
	
97
	return startingWith + " (" + QString::number(playlistNumber) + ")";	
98 99 100 101 102 103
    }
    else
    {
	while(names.contains(startingWith + ' ' + QString::number(playlistNumber)) != 0)
	    playlistNumber++;
	
104
	return startingWith + " " + QString::number(playlistNumber);
105
    }
106 107
}

108
QString PlaylistSplitter::playNextFile(bool random, bool loopPlaylist)
109
{
110 111
    PlaylistItem *i;

112 113 114
    // Four basic cases here:  (1) We've asked for a specific next item, (2) play
    // the item that's after the currently playing item, (3) play the selected 
    // item or (4) play the first item in the list.
115

116
    if(m_nextPlaylistItem && m_nextPlaylistItem != m_playingItem) {
117 118 119
        i = m_nextPlaylistItem;
        m_nextPlaylistItem = 0;
    }
120
    else if(m_playingItem) {
121 122
        Playlist *p = static_cast<Playlist *>(m_playingItem->listView());
        i = p->nextItem(m_playingItem, random);
123
        if(!i && loopPlaylist)
124
            i = static_cast<PlaylistItem *>(p->firstChild());
125 126
    }
    else {
127
        i = playlistSelection().first();
128 129
        if(!i)
            i = static_cast<PlaylistItem *>(visiblePlaylist()->firstChild());
130 131
    }

132
    return play(i);
133 134
}

135
QString PlaylistSplitter::playPreviousFile(bool random)
136
{
137
    if(!m_playingItem)
138
	return QString::null;
139

140 141
    Playlist *p = static_cast<Playlist *>(m_playingItem->listView());
    PlaylistItem *i = p->previousItem(m_playingItem, random);
142

143
    return play(i);
144 145
}

146 147 148 149 150 151 152 153
QString PlaylistSplitter::playSelectedFile()
{
    if(playlistSelection().isEmpty())
	return QString::null;
    else
	return play(playlistSelection().first());
}

154 155 156 157
QString PlaylistSplitter::playFirstFile()
{
    Playlist *p = visiblePlaylist();
    PlaylistItem *i = static_cast<PlaylistItem *>(p->firstChild());
158

159
    return play(i);
160 161
}

162 163
void PlaylistSplitter::stop()
{
164 165
    m_nextPlaylistItem = 0;

166 167 168 169 170
    if(!m_playingItem)
	return;

    Playlist *p = static_cast<Playlist *>(m_playingItem->listView());

171 172 173
    if(p)
	p->setPlaying(m_playingItem, false);

174
    m_playingItem = 0;
175 176 177 178
}

QString PlaylistSplitter::playingArtist() const
{
179 180
    if(m_playingItem)
	return m_playingItem->text(PlaylistItem::ArtistColumn);
181
    else
182
	return QString::null;
183 184 185 186
}

QString PlaylistSplitter::playingTrack() const
{
187 188
    if(m_playingItem)
	return m_playingItem->text(PlaylistItem::TrackColumn);
189
    else
190
	return QString::null;
191 192 193 194
}

QString PlaylistSplitter::playingList() const
{
195 196
    if(m_playingItem)
	return static_cast<Playlist *>(m_playingItem->listView())->name();
197
    else
198
	return QString::null;
199 200
}

201
QString PlaylistSplitter::extensionsString(const QStringList &extensions, const QString &type) // static
202
{
203 204 205 206 207 208 209 210 211 212 213 214
    QStringList l;

    for(QStringList::ConstIterator it = extensions.begin(); it != extensions.end(); ++it)
	l.append(QString("*." + (*it)));

    // i.e. "*.m3u, *.mp3|Media Files"

    QString s = l.join(" ");

    if(type != QString::null)
	s += "|" + type + " (" + l.join(", ") + ")";

215
    return s;
216 217
}

218 219
void PlaylistSplitter::open(const QString &file) 
{
220 221 222
    if(file.isEmpty())
	return;

223
    if(visiblePlaylist() == m_collection || 
224 225 226 227 228 229
       KMessageBox::questionYesNo(this, 
				  i18n("Do you want to add this item to the current list or to the collection list?"), 
				  QString::null, 
				  KGuiItem(i18n("Current")), 
				  KGuiItem(i18n("Collection"))) == KMessageBox::No)
    {
230
	slotAddToPlaylist(file, m_collection);
231
    }
232
    else
233
	slotAddToPlaylist(file, visiblePlaylist());
234 235 236 237
}

void PlaylistSplitter::open(const QStringList &files) 
{
238 239 240
    if(files.isEmpty())
	return;
    
241
    if(visiblePlaylist() == m_collection || 
242 243 244 245 246 247
       KMessageBox::questionYesNo(this, 
				  i18n("Do you want to add these items to the current list or to the collection list?"), 
				  QString::null, 
				  KGuiItem(i18n("Current")), 
				  KGuiItem(i18n("Collection"))) == KMessageBox::No) 
    {
248
	slotAddToPlaylist(files, m_collection);
249
    }
250
    else
251
	slotAddToPlaylist(files, visiblePlaylist());
252 253 254 255
}

Playlist *PlaylistSplitter::createPlaylist(const QString &name)
{
256
    Playlist *p = new Playlist(m_playlistStack, name.latin1());
257 258
    setupPlaylist(p, true);
    return p;
259 260
}

261 262 263 264
////////////////////////////////////////////////////////////////////////////////
// public slots
////////////////////////////////////////////////////////////////////////////////

265
void PlaylistSplitter::slotOpen()
266
{
267 268
    QStringList files = KFileDialog::getOpenFileNames(
	QString::null, extensionsString((*m_mediaExtensions + *m_listExtensions), i18n("Media Files")));
269 270 271
    open(files);
}

272
void PlaylistSplitter::slotOpenDirectory()
273
{ 
274
    DirectoryList *l = new DirectoryList(m_directoryList, this, "directoryList");
275

276
    m_directoryQueue.clear();
277
    m_directoryQueueRemove.clear();
278

279 280 281 282
    connect(l, SIGNAL(signalDirectoryAdded(const QString &)), 
	    this, SLOT(slotQueueDirectory(const QString &)));
    connect(l, SIGNAL(signalDirectoryRemoved(const QString &)), 
	    this, SLOT(slotQueueDirectoryRemove(const QString &)));
283 284

    if(l->exec() == QDialog::Accepted) {
285
	open(m_directoryQueue);
286 287 288
	for(QStringList::Iterator it = m_directoryQueue.begin(); it !=  m_directoryQueue.end(); it++)
	    m_dirWatch->addDir(*it, false, true);
	    
289
	m_directoryList += m_directoryQueue;
290 291 292

	QStringList::Iterator it = m_directoryQueueRemove.begin();
	for(; it !=  m_directoryQueueRemove.end(); it++) {
293
	    m_dirWatch->removeDir(*it);
294
	    m_directoryList.remove(*it);
295
	}
296 297
    }
}
298

299
Playlist *PlaylistSplitter::slotCreatePlaylist()
300 301 302
{
    bool ok;

303
    // If this text is changed, please also change it in PlaylistBox::duplicate().
304

Nadeem Hasan's avatar
Nadeem Hasan committed
305 306 307
    QString name = KLineEditDlg::getText(i18n("Create New Playlist"), 
	i18n("Please enter a name for the new playlist:"),
	uniquePlaylistName(), &ok);
308
    if(ok)
309
	return createPlaylist(name);
310
    else
311
	return 0;
312 313
}

314
void PlaylistSplitter::slotSelectPlaying()
315
{
316
    if(!m_playingItem)
317 318
	return;

319
    Playlist *l = static_cast<Playlist *>(m_playingItem->listView());
320
	
321 322
    if(!l)
	return;
323

324
    l->clearSelection();
325 326
    l->setSelected(m_playingItem, true);
    l->ensureItemVisible(m_playingItem);
Nadeem Hasan's avatar
Nadeem Hasan committed
327

328 329
    if(l != visiblePlaylist())
	m_playlistBox->raise(l);
330 331
}

332
void PlaylistSplitter::slotDeleteSelectedItems()
333 334 335
{
    Playlist *p = visiblePlaylist();
    if(p)
336
	p->slotDeleteSelectedItems();
337 338
}

339 340 341 342 343
void PlaylistSplitter::slotAddToPlaylist(const QString &file, Playlist *list)
{
    KApplication::setOverrideCursor(Qt::waitCursor);
    addImpl(file, list);
    KApplication::restoreOverrideCursor();
Nadeem Hasan's avatar
Nadeem Hasan committed
344

345 346 347 348 349 350 351 352 353 354 355 356 357 358 359
    if(m_editor)
	m_editor->slotUpdateCollection();
}

void PlaylistSplitter::slotAddToPlaylist(const QStringList &files, Playlist *list)
{
    KApplication::setOverrideCursor(Qt::waitCursor);
    for(QStringList::ConstIterator it = files.begin(); it != files.end(); ++it)
        addImpl(*it, list);
    KApplication::restoreOverrideCursor();

    if(m_editor)
	m_editor->slotUpdateCollection();
}

360 361 362 363 364 365 366
void PlaylistSplitter::slotGuessTagInfo()
{
    visiblePlaylist()->slotGuessTagInfo();
    if(m_editor)
        m_editor->slotRefresh();
}

367 368 369 370 371 372 373
void PlaylistSplitter::slotRenameFile()
{
    visiblePlaylist()->slotRenameFile();
    if(m_editor)
        m_editor->slotRefresh();
}

374 375 376 377 378 379
////////////////////////////////////////////////////////////////////////////////
// private members
////////////////////////////////////////////////////////////////////////////////

void PlaylistSplitter::setupLayout()
{
380
    m_playlistBox = new PlaylistBox(this, "playlistBox");
381 382 383 384 385 386

    // Create a splitter to go between the playlists and the editor.

    QSplitter *editorSplitter = new QSplitter(Qt::Vertical, this, "editorSplitter");

    // Create the playlist and the editor.
387
    
388 389
    m_playlistStack = new QWidgetStack(editorSplitter, "playlistStack");
    m_editor = new TagEditor(editorSplitter, "tagEditor");
390 391 392

    // Make the editor as small as possible (or at least as small as recommended)

393
    editorSplitter->setResizeMode(m_editor, QSplitter::FollowSizeHint);
394 395 396 397

    // Make the connection that will update the selected playlist when a 
    // selection is made in the playlist box.

398 399
    connect(m_playlistBox, SIGNAL(signalCurrentChanged(const PlaylistList &)), 
	    this, SLOT(slotChangePlaylist(const PlaylistList &)));
400

401
    connect(m_playlistBox, SIGNAL(signalDoubleClicked()), this, SIGNAL(signalListBoxDoubleClicked()));
402 403 404 405

    // Create the collection list; this should always exist.  This has a 
    // slightly different creation process than normal playlists (since it in
    // fact is a subclass) so it is created here rather than by using 
406
    // slotCreatePlaylist().
407

408
    CollectionList::initialize(m_playlistStack, m_restore);
409 410
    m_collection = CollectionList::instance();
    setupPlaylist(m_collection, true, "folder_sound");
411
    connect(m_collection, SIGNAL(signalCollectionChanged()), m_editor, SLOT(slotUpdateCollection()));
412 413
    connect(m_collection, SIGNAL(signalRequestPlaylistCreation(const PlaylistItemList &)), 
	    this, SLOT(slotCreatePlaylist(const PlaylistItemList &)));
414

415 416 417 418 419 420 421 422 423

    // Create the search widget -- this must be done after the CollectionList is created.
    m_searchWidget = new SearchWidget(editorSplitter, CollectionList::instance(), "searchWidget");
    editorSplitter->moveToFirst(m_searchWidget);
    connect(m_searchWidget, SIGNAL(signalQueryChanged(const QString &, bool)),
	    this, SLOT(slotShowSearchResults(const QString &, bool)));
    connect(CollectionList::instance(), SIGNAL(signalVisibleColumnsChanged()),
	    this, SLOT(slotVisibleColumnsChanged()));

424
    // Show the collection on startup.
425
    m_playlistBox->setSelected(0, true);
426 427 428 429
}

void PlaylistSplitter::readConfig()
{
430 431 432 433
    KConfig *config = KGlobal::config();
    { // block for Playlists group
	KConfigGroupSaver saver(config, "Playlists");

434 435 436 437 438 439 440
	QValueList<int> splitterSizes = config->readIntListEntry("PlaylistSplitterSizes");
	if(splitterSizes.isEmpty()) {
	    splitterSizes.append(100);
	    splitterSizes.append(640);
	}
	setSizes(splitterSizes);
	
441
	if(m_restore) {
442 443 444 445

	    QString playlistsFile = KGlobal::dirs()->saveLocation("appdata") + "playlists";

	    QFile f(playlistsFile);
Nadeem Hasan's avatar
Nadeem Hasan committed
446

447 448 449
	    if(f.open(IO_ReadOnly)) {
		QDataStream s(&f);
		while(!s.atEnd()) {
450
		    Playlist *p = new Playlist(m_playlistStack);
451 452 453 454
		    s >> *p;

		    // check to see if we've alredy loaded this item before continuing

455
		    if(p->fileName().isEmpty() || !m_playlistFiles.insert(p->fileName()))
456 457 458 459
			setupPlaylist(p);
		    else
			delete p;
		}
460
	    }
461

462
	    m_directoryList = config->readListEntry("DirectoryList");
463
	    QTimer::singleShot(0, this, SLOT(slotScanDirectories()));
464 465

	    m_dirWatch = new KDirWatch();
466 467
	    connect(m_dirWatch, SIGNAL(dirty(const QString &)), 
		    this, SLOT(slotDirChanged(const QString &)));
468

469 470
	    QStringList::Iterator it = m_directoryList.begin();
            for(; it != m_directoryList.end(); ++it)
471 472 473
		m_dirWatch->addDir(*it, false, true);

	    m_dirWatch->startScan();
474
	}
475 476 477

	// restore the list of hidden and shown columns

478
	if(m_collection) {
479
	    for(int i = 0; i < m_collection->columns(); i++)
480
		m_columnNames.append(m_collection->columnText(i));
481
	}
482

483
    }
484
}
485 486 487 488

void PlaylistSplitter::saveConfig()
{
    KConfig *config = KGlobal::config();
489

490
    // Save the list of open playlists.
Nadeem Hasan's avatar
Nadeem Hasan committed
491

492
    if(m_restore && m_playlistBox) {
493 494 495

	// Start at item 1.  We want to skip the collection list.

496 497 498 499 500 501 502
	QString playlistsFile = KGlobal::dirs()->saveLocation("appdata") + "playlists";
	QFile f(playlistsFile);
	
	if(f.open(IO_WriteOnly)) {

	    QDataStream s(&f);

503
	    PlaylistList l = m_playlistBox->playlists();
504

505 506
	    for(PlaylistList::Iterator it = l.begin(); it != l.end(); it++)
		s << *(*it);
507

508
	    f.close();
509 510 511
	}
	{ // block for Playlists group
	    KConfigGroupSaver saver(config, "Playlists");
512
	    config->writeEntry("DirectoryList", m_directoryList);
513
	    config->writeEntry("SortColumn", m_collection->sortColumn());
514
	    config->writeEntry("PlaylistSplitterSizes", sizes());
515 516
	}
    }
517 518
}

519 520 521 522 523 524 525 526 527 528 529 530 531
void PlaylistSplitter::addImpl(const QString &file, Playlist *list)
{
    processEvents();
    QFileInfo fileInfo(QDir::cleanDirPath(file));
    if(fileInfo.exists()) {
        if(fileInfo.isDir()) {
            QDir dir(fileInfo.filePath());
            QStringList dirContents=dir.entryList();
            for(QStringList::Iterator it = dirContents.begin(); it != dirContents.end(); ++it)
                if(*it != "." && *it != "..")
                    addImpl(fileInfo.filePath() + QDir::separator() + *it, list);
        }
        else {
532
            QString extension = fileInfo.extension(false).lower();
533
            if(m_mediaExtensions->contains(extension) > 0)
534
		list->createItem(fileInfo);
535
	    else if(m_listExtensions->contains(extension) > 0)
536 537
		openPlaylist(fileInfo.absFilePath());
        }
Nadeem Hasan's avatar
Nadeem Hasan committed
538
    }
539 540
}

541
void PlaylistSplitter::setupPlaylist(Playlist *p, bool raise, const char *icon)
542
{
543 544 545 546 547 548 549 550 551 552 553 554
    connect(p, SIGNAL(signalSelectionChanged(const PlaylistItemList &)),
	    m_editor, SLOT(slotSetItems(const PlaylistItemList &)));
    connect(p, SIGNAL(signalDoubleClicked()),
	    this, SIGNAL(signalDoubleClicked()));
    connect(p, SIGNAL(signalNumberOfItemsChanged(Playlist *)),
	    this, SLOT(slotPlaylistCountChanged(Playlist *)));
    connect(p, SIGNAL(signalAboutToRemove(PlaylistItem *)), 
	    this, SLOT(slotPlaylistItemRemoved(PlaylistItem *)));
    connect(p, SIGNAL(signalFilesDropped(const QStringList &, Playlist *)), 
	    this, SLOT(slotAddToPlaylist(const QStringList &, Playlist *)));
    connect(p, SIGNAL(signalSetNext(PlaylistItem *)),
	    this, SLOT(slotSetNextItem(PlaylistItem *)));
555

556 557
    if(icon)
	m_playlistBox->createItem(p, icon, raise);
558

559
    if(raise)
560
	m_playlistStack->raiseWidget(p);
561 562
}

563 564 565
Playlist *PlaylistSplitter::openPlaylist(const QString &file)
{
    QFileInfo fileInfo(file);
566 567 568 569 570
    if(!fileInfo.exists() || 
       !fileInfo.isFile() || 
       !fileInfo.isReadable() || 
       m_playlistFiles.insert(fileInfo.absFilePath()))
    {
571
	return 0;
572
    }
573

574
    Playlist *p = new Playlist(file, m_playlistStack, fileInfo.baseName(true).latin1());
575
    setupPlaylist(p);
576
    return p;
577 578
}

579 580 581 582 583 584 585
QString PlaylistSplitter::play(PlaylistItem *item)
{
    stop();

    if(!item)
	return QString::null;

586 587 588 589
    Playlist *p = static_cast<Playlist *>(item->listView());

    p->setPlaying(item, true);

590 591 592 593 594
    m_playingItem = item;

    return item->absFilePath();
}

595 596 597 598
////////////////////////////////////////////////////////////////////////////////
// private slots
////////////////////////////////////////////////////////////////////////////////

599
void PlaylistSplitter::slotChangePlaylist(const PlaylistList &l)
600
{
601
    if(l.isEmpty())
602
	return;
603

604 605
    Playlist *current = m_dynamicList;

606 607
    if(m_searchWidget)
	m_searchWidget->clear();
608 609

    m_nextPlaylistItem = 0;
610 611 612 613 614 615 616 617 618 619 620 621 622
    if(l.count() == 1) {
	m_playlistStack->raiseWidget(l.first());
	m_editor->slotSetItems(playlistSelection());
	m_dynamicList = 0;
	emit signalPlaylistChanged();
    }
    else {
	m_dynamicList = new Playlist(m_playlistStack, i18n("Dynamic List"));
	setupPlaylist(m_dynamicList, true, 0);
	// unsorted...
	m_dynamicList->setSorting(m_dynamicList->columns() + 1);

	for(PlaylistList::ConstIterator it = l.begin(); it != l.end(); ++it) {
623 624
	    if(*it)
		m_dynamicList->createItems((*it)->items());
625 626 627 628 629
	}
    }

    if(current)
	delete current;
630 631
}

632
void PlaylistSplitter::slotPlaylistCountChanged(Playlist *p)
633
{
634 635
    if(p && p == m_playlistStack->visibleWidget())
	emit signalSelectedPlaylistCountChanged(p->childCount());
636 637
}

638
void PlaylistSplitter::slotPlaylistItemRemoved(PlaylistItem *item)
639
{
640 641
    if(item == m_playingItem)
	m_playingItem = 0;
642 643 644

    if(item == m_nextPlaylistItem)
	m_nextPlaylistItem = 0;
645 646
}

647
void PlaylistSplitter::slotCreatePlaylist(const PlaylistItemList &items)
648
{
649 650 651
    if(items.isEmpty())
	return;

652
    Playlist *playlist = slotCreatePlaylist();
653

654
    if(!playlist)
655
        return;
656

657
    playlist->createItems(items);
658 659
}

660 661 662 663 664 665 666 667 668 669
void PlaylistSplitter::slotShowSearchResults(const QString &query, bool caseSensitive)
{
    if(query.isEmpty()) {
	visiblePlaylist()->setItemsVisible(visiblePlaylist()->items(), true);
	return;
    }

    PlaylistList playlists;
    playlists.append(visiblePlaylist());

670
    PlaylistSearch::Component component(query, caseSensitive, m_searchWidget->searchedColumns(0));
671 672 673 674 675 676 677 678 679
    PlaylistSearch::ComponentList components;
    components.append(&component);

    PlaylistSearch search(playlists, components);

    Playlist::setItemsVisible(search.matchedItems(), true);    
    Playlist::setItemsVisible(search.unmatchedItems(), false);
}

680 681 682 683 684 685 686 687
void PlaylistSplitter::slotVisibleColumnsChanged()
{
    m_searchWidget->slotUpdateColumns();
    m_searchWidget->slotQueryChanged();
    if(m_searchWidget->searchedColumns(0).count() > 1)
	slotShowSearchResults(m_searchWidget->query(), m_searchWidget->caseSensitive());
}

688
#include "playlistsplitter.moc"
Nadeem Hasan's avatar
Nadeem Hasan committed
689 690

// vim:ts=8