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.
tdegraphics/kpovmodeler/pmpovrayrenderwidget.cpp

438 lines
12 KiB

/*
**************************************************************************
description
--------------------
copyright : (C) 2001-2003 by Andreas Zehender
email : zehender@kde.org
**************************************************************************
**************************************************************************
* *
* 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. *
* *
**************************************************************************/
#include "pmpovrayrenderwidget.h"
#include "pmdefaults.h"
#include "pmdebug.h"
#include "pmdragwidget.h"
#include <kprocess.h>
#include <kmessagebox.h>
#include <klocale.h>
#include <kconfig.h>
#include <kurl.h>
#include <ktempfile.h>
#include <qcolor.h>
#include <qpainter.h>
#include <qtextstream.h>
#include <qdragobject.h>
#ifdef KDE_NO_COMPAT
#undef KDE_NO_COMPAT
#endif
#include <kapplication.h>
QString PMPovrayRenderWidget::s_povrayCommand = c_defaultPovrayCommand;
QStringList PMPovrayRenderWidget::s_libraryPaths;
PMPovrayRenderWidget::PMPovrayRenderWidget( QWidget* parent, const char* name )
: PMDragWidget( parent, name )
{
m_pProcess = 0;
m_bSuspended = false;
m_rcvHeader = false;
m_skipBytes = 0;
m_bPixmapUpToDate = false;
m_pTempFile = 0;
setBackgroundColor( QColor( 0, 0, 0 ) );
}
PMPovrayRenderWidget::~PMPovrayRenderWidget( )
{
cleanup( );
}
bool PMPovrayRenderWidget::render( const QByteArray& scene,
const PMRenderMode& m,
const KURL& documentURL )
{
cleanup( );
m_povrayOutput = "";
m_renderMode = m;
if( !scene.data( ) )
{
KMessageBox::sorry( this, i18n( "Can't render an empty scene.\n" ) );
return false;
}
// output to tmp file
m_pTempFile = new KTempFile( QString::null, ".pov" );
QDataStream* dstr = m_pTempFile->dataStream( );
if( ( m_pTempFile->status( ) != 0 ) || !dstr )
{
KMessageBox::sorry( this, i18n( "Couldn't write the scene to a temp file.\n" ) );
return false;
}
dstr->writeRawBytes( scene.data( ), scene.size( ) );
m_pTempFile->close( );
m_pProcess = new KProcess( );
connect( m_pProcess, SIGNAL( receivedStdout( KProcess*, char*, int ) ),
SLOT( slotPovrayImage( KProcess*, char*, int ) ) );
connect( m_pProcess, SIGNAL( receivedStderr( KProcess*, char*, int ) ),
SLOT( slotPovrayMessage( KProcess*, char*, int ) ) );
connect( m_pProcess, SIGNAL( processExited( KProcess* ) ),
SLOT( slotRenderingFinished( KProcess* ) ) );
*m_pProcess << s_povrayCommand;
QStringList::ConstIterator it;
QStringList args = m_renderMode.commandLineSwitches( );
for( it = args.begin( ); it != args.end( ); ++it )
*m_pProcess << *it;
for( it = s_libraryPaths.begin( ); it != s_libraryPaths.end( ); ++it )
{
QString path = *it;
if( path != QString( "/" ) )
if( path.right( 1 ) == QString( "/" ) )
path.truncate( path.length( ) - 1 );
*m_pProcess << ( QString( "+L" ) + path );
}
*m_pProcess << QString( "+I" ) + m_pTempFile->name( ) << "+O-" << "+FT"
<< "+K0.0" << "+KFI1" << "+KFF1" << "+KI0.0" << "+KF0.0"
<< "+SF1" << "+EF1" << "-KC" << "-D";
#if ( ( KDE_VERSION_MAJOR == 2 ) && ( KDE_VERSION_MINOR >= 9 ) ) || ( KDE_VERSION_MAJOR == 3 )
if( !documentURL.isEmpty( ) && documentURL.isLocalFile( ) )
m_pProcess->setWorkingDirectory( documentURL.directory( ) );
#endif
m_rcvHeader = true;
m_rcvHeaderBytes = 0;
m_rcvPixels = 0;
m_progress = 0;
m_numRestBytes = 0;
m_line = 0;
m_column = 0;
m_skipBytes = 0;
int width = m_renderMode.width( );
int height = m_renderMode.height( );
m_image.create( width, height, 32 );
m_image.setAlphaBuffer( m_renderMode.alpha( ) );
m_image.fill( qRgb( 0, 0, 0 ) );
m_bPixmapUpToDate = false;
repaint( );
if( !m_pProcess->start( KProcess::NotifyOnExit, KProcess::AllOutput ) )
{
KMessageBox::error( this, i18n( "Couldn't call povray.\n"
"Please check your installation "
"or set another povray command." ) );
delete m_pProcess;
m_pProcess = 0;
return false;
}
m_bSuspended = false;
return true;
}
void PMPovrayRenderWidget::killRendering( )
{
if( m_pProcess )
{
if( m_bSuspended )
m_pProcess->kill( SIGCONT );
m_bSuspended = false;
m_pProcess->kill( );
}
}
void PMPovrayRenderWidget::suspendRendering( )
{
if( m_pProcess )
{
m_bSuspended = true;
m_pProcess->kill( SIGSTOP );
}
}
void PMPovrayRenderWidget::resumeRendering( )
{
if( m_pProcess )
{
m_pProcess->kill( SIGCONT );
m_bSuspended = false;
}
}
void PMPovrayRenderWidget::slotPovrayMessage( KProcess*,
char* buffer, int buflen )
{
QString str;
str.setLatin1( buffer, buflen );
m_povrayOutput += str;
emit povrayMessage( str );
}
void PMPovrayRenderWidget::slotPovrayImage( KProcess*, char* buffer, int buflen )
{
int index = 0;
int i;
int oldLine = m_line;
if( m_rcvHeader )
{
// receive targa header
while( ( m_rcvHeaderBytes < 18 ) && ( index < buflen ) )
{
m_header[m_rcvHeaderBytes] = ( unsigned char ) buffer[index];
m_rcvHeaderBytes++;
index++;
}
if( m_rcvHeaderBytes == 18 )
{
// complete targa header received
m_rcvHeader = false;
m_skipBytes = m_header[0]; // id length
m_bytespp = m_header[16] / 8;
}
}
if( m_skipBytes > 0 )
{
int skip = buflen - index;
if( skip > m_skipBytes )
skip = m_skipBytes;
m_skipBytes -= skip;
index += skip;
}
if( ( m_numRestBytes > 0 ) && ( index < buflen ) )
{
while( ( m_numRestBytes < m_bytespp ) && ( index < buflen ) )
{
m_restBytes[m_numRestBytes] = ( unsigned char ) buffer[index];
index++;
m_numRestBytes++;
}
if( m_numRestBytes == m_bytespp )
{
m_numRestBytes = 0;
if( m_bytespp == 4 )
setPixel( m_column, m_line,
qRgba( m_restBytes[2], m_restBytes[1],
m_restBytes[0], m_restBytes[3] ) );
else
setPixel( m_column, m_line,
qRgb( m_restBytes[2], m_restBytes[1], m_restBytes[0] ) );
m_column++;
m_rcvPixels++;
if( m_column == m_renderMode.width( ) )
{
m_column = 0;
m_line++;
}
}
}
if( index < buflen )
{
int num = ( buflen - index ) / m_bytespp;
for( i = 0; i < num; i++ )
{
if( m_bytespp == 4 )
setPixel( m_column, m_line,
qRgba( buffer[index+2], buffer[index+1],
buffer[index], buffer[index+3] ) );
else
setPixel( m_column, m_line,
qRgb( buffer[index+2], buffer[index+1],
buffer[index] ) );
index += m_bytespp;
m_column++;
m_rcvPixels++;
if( m_column == m_renderMode.width( ) )
{
m_column = 0;
m_line++;
}
}
}
if( index < buflen )
{
m_numRestBytes = buflen - index;
for( i = 0; i < m_numRestBytes; i++ )
{
m_restBytes[i] = buffer[index];
index++;
}
}
if( m_line != oldLine )
{
QPainter paint( this );
int offset = 0;
if( m_renderMode.subSection( ) )
{
double sr = m_renderMode.startRow( );
if( sr < 1 )
offset = ( int ) ( m_renderMode.height( ) * sr + 0.5 );
else
offset += ( int ) sr;
}
paint.drawImage( 0, offset + oldLine,
m_image.copy( 0, offset + oldLine, m_image.width( ), offset + m_line - oldLine ) );
emit lineFinished( m_line - 1 );
}
int oldProgress = m_progress;
int numPixels = 0;
if( m_renderMode.subSection( ) )
{
int sr = 0;
if( m_renderMode.startRow( ) < 1 )
sr = ( int ) ( m_renderMode.height( ) * m_renderMode.startRow( ) + 0.5 );
else
sr = ( int ) m_renderMode.startRow( );
int er = 0;
if( m_renderMode.endRow( ) < 1 )
er = ( int ) ( m_renderMode.height( ) * m_renderMode.endRow( ) + 0.5 );
else
er = ( int ) m_renderMode.endRow( );
numPixels = m_renderMode.width( ) * ( er - sr );
}
else
numPixels = m_renderMode.width( ) * m_renderMode.height( );
m_progress = m_rcvPixels * 100 / numPixels;
if( m_progress != oldProgress )
emit progress( m_progress );
m_bPixmapUpToDate = false;
}
void PMPovrayRenderWidget::setPixel( int x, int y, uint c )
{
if( m_renderMode.subSection( ) )
{
double sr = m_renderMode.startRow( );
if( sr < 1 )
y += ( int ) ( m_renderMode.height( ) * sr + 0.5 );
else
y += ( int ) sr;
}
if( x >= 0 && x < m_image.width( ) &&
y >= 0 && y < m_image.height( ) )
m_image.setPixel( x, y, c );
}
/**
void PMPovrayRenderWidget::slotWroteStdin( KProcess* )
{
if( m_pProcess )
m_pProcess->closeStdin( );
m_data.resize( 0 );
}
*/
void PMPovrayRenderWidget::slotRenderingFinished( KProcess* )
{
if( m_pProcess->normalExit( ) )
emit( finished( m_pProcess->exitStatus( ) ) );
else
emit( finished( -1000 ) );
cleanup( );
}
void PMPovrayRenderWidget::paintEvent( QPaintEvent* ev )
{
if( !m_bPixmapUpToDate )
{
if( !m_image.isNull( ) )
m_pixmap.convertFromImage( m_image );
m_bPixmapUpToDate = true;
}
bitBlt( this, ev->rect( ).left( ), ev->rect( ).top( ),
&m_pixmap, ev->rect( ).left( ), ev->rect( ).top( ),
ev->rect( ).width( ), ev->rect( ).height( ), CopyROP );
}
void PMPovrayRenderWidget::cleanup( )
{
if( m_pProcess )
delete m_pProcess;
m_pProcess = 0;
if( m_pTempFile )
{
m_pTempFile->unlink( );
delete m_pTempFile;
}
m_pTempFile = 0;
}
QSize PMPovrayRenderWidget::sizeHint( ) const
{
QSize s;
if( m_image.isNull( ) )
s = QSize( 200, 200 );
else
s = m_image.size( );
return s.expandedTo( minimumSize( ) );
}
void PMPovrayRenderWidget::saveConfig( KConfig* cfg )
{
cfg->setGroup( "Povray" );
#if ( ( KDE_VERSION_MAJOR == 3 ) && ( KDE_VERSION_MINOR <= 1 ) )
cfg->writeEntry( "PovrayCommand", s_povrayCommand );
cfg->writeEntry( "LibraryPaths", s_libraryPaths );
#else
cfg->writePathEntry( "PovrayCommand", s_povrayCommand );
cfg->writePathEntry( "LibraryPaths", s_libraryPaths );
#endif
}
void PMPovrayRenderWidget::restoreConfig( KConfig* cfg )
{
cfg->setGroup( "Povray" );
#if ( ( KDE_VERSION_MAJOR == 3 ) && ( KDE_VERSION_MINOR <= 1 ) )
s_povrayCommand = cfg->readEntry( "PovrayCommand", s_povrayCommand );
s_libraryPaths = cfg->readListEntry( "LibraryPaths" );
#else
s_povrayCommand = cfg->readPathEntry( "PovrayCommand", s_povrayCommand );
s_libraryPaths = cfg->readPathListEntry( "LibraryPaths" );
#endif
}
void PMPovrayRenderWidget::startDrag( )
{
QImageDrag* d = new QImageDrag( m_image, this );
d->dragCopy( );
}
#include "pmpovrayrenderwidget.moc"