summaryrefslogtreecommitdiffstats
path: root/src/value.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/value.cpp')
-rw-r--r--src/value.cpp465
1 files changed, 465 insertions, 0 deletions
diff --git a/src/value.cpp b/src/value.cpp
new file mode 100644
index 0000000..c31514e
--- /dev/null
+++ b/src/value.cpp
@@ -0,0 +1,465 @@
+/***************************************************************************
+* Copyright (C) 2004-2009 by Thomas Fischer *
+* fischer@unix-ag.uni-kl.de *
+* *
+* 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. *
+* *
+* 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., *
+* 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
+***************************************************************************/
+#include <qregexp.h>
+#include <qstring.h>
+#include <qstringlist.h>
+
+#include "value.h"
+
+namespace BibTeX
+{
+
+ ValueTextInterface::ValueTextInterface( const ValueTextInterface* other ) : m_text( other->text() )
+ {
+// nothing
+ }
+
+ ValueTextInterface::ValueTextInterface( const QString& text ) : m_text( text )
+ {
+// nothing
+ }
+
+ void ValueTextInterface::setText( const QString& text )
+ {
+ m_text = text;
+ }
+
+ QString ValueTextInterface::text() const
+ {
+ return m_text;
+ }
+
+ QString ValueTextInterface::simplifiedText() const
+ {
+ return text().replace( QRegExp( "\\\\[A-Za-z0-9]+" ), "" ).replace( '{', "" ).replace( '}', "" );
+ }
+
+ void ValueTextInterface::replace( const QString &before, const QString &after )
+ {
+ if ( before == text() || before == simplifiedText() )
+ setText( after );
+ }
+
+ bool ValueTextInterface::containsPattern( const QString &pattern, bool caseSensitive )
+ {
+ return text().contains( pattern, caseSensitive ) || simplifiedText().contains( pattern, caseSensitive );
+ }
+
+ ValueItem::ValueItem( const QString& text ) : ValueTextInterface( text )
+ {
+// nothing
+ }
+
+ Keyword::Keyword( Keyword *other ) : ValueTextInterface( other )
+ {
+// nothing
+ }
+
+ Keyword::Keyword( const QString& text ) : ValueTextInterface( text )
+ {
+// nothing
+ }
+
+ Keyword *Keyword::clone()
+ {
+ return new Keyword( text() );
+ }
+
+ KeywordContainer::KeywordContainer(): ValueItem( "" )
+ {
+// nothing
+ }
+
+ KeywordContainer::KeywordContainer( const QString& text ) : ValueItem( text )
+ {
+ setText( text );
+ }
+
+ KeywordContainer::KeywordContainer( KeywordContainer *other ) : ValueItem( QString::null )
+ {
+ for ( QValueList<Keyword*>::Iterator it = other->keywords.begin(); it != other->keywords.end(); ++it )
+ keywords.append(( *it )->clone() );
+ }
+
+ KeywordContainer::KeywordContainer( const QStringList& list ) : ValueItem( QString::null )
+ {
+ setList( list );
+ }
+
+ ValueItem *KeywordContainer::clone()
+ {
+ return new KeywordContainer( this );
+ }
+
+ void KeywordContainer::setList( const QStringList& list )
+ {
+ keywords.clear();
+ for ( QStringList::ConstIterator it = list.begin(); it != list.end(); ++it )
+ keywords.append( new Keyword( *it ) );
+ }
+
+ void KeywordContainer::append( const QString& text )
+ {
+ bool contains = FALSE;
+
+ for ( QValueList<Keyword*>::ConstIterator it = keywords.begin(); !contains && it != keywords.end(); ++it )
+ contains = QString::compare(( *it )->text(), text ) == 0;
+
+ if ( contains == 0 )
+ keywords.append( new Keyword( text ) );
+ }
+
+ void KeywordContainer::remove( const QString& text )
+ {
+ bool contains = FALSE;
+ for ( QValueList<Keyword*>::Iterator it = keywords.begin(); !contains && it != keywords.end(); ++it )
+ if ( QString::compare(( *it )->text(), text ) == 0 )
+ {
+ keywords.remove( it );
+ break;
+ }
+ }
+
+ void KeywordContainer::setText( const QString& text )
+ {
+ ValueItem::setText( text );
+
+ QRegExp splitRegExp;
+ if ( text.contains( ";" ) )
+ splitRegExp = QRegExp( "\\s*;\\s*" );
+ else
+ splitRegExp = QRegExp( "\\s*,\\s*" );
+
+ keywords.clear();
+ QStringList keywordList = QStringList::split( splitRegExp, text, FALSE );
+ for ( QStringList::ConstIterator it = keywordList.begin(); it != keywordList.end(); ++it )
+ keywords.append( new Keyword( *it ) );
+ }
+
+ QString KeywordContainer::text() const
+ {
+ QString result;
+ bool first = true;
+ for ( QValueList<Keyword*>::ConstIterator it = keywords.begin(); it != keywords.end(); ++it )
+ {
+ if ( !first )
+ result.append( "; " );
+ else first = false;
+ result.append(( *it )->text() );
+ }
+ return result;
+ }
+
+ void KeywordContainer::replace( const QString &before, const QString &after )
+ {
+ for ( QValueList<Keyword*>::ConstIterator it = keywords.begin(); it != keywords.end(); ++it )
+ ( *it )->replace( before, after );
+ }
+
+ Person::Person( const QString& text, bool firstNameFirst ) : ValueTextInterface( text ), m_firstNameFirst( firstNameFirst )
+ {
+ setText( text );
+ }
+
+ Person::Person( const QString& firstName, const QString& lastName, bool firstNameFirst ) : ValueTextInterface( QString( firstName ).append( " " ).append( lastName ) ), m_firstName( firstName ), m_lastName( lastName ), m_firstNameFirst( firstNameFirst )
+ {
+// nothing
+ }
+
+ Person *Person::clone()
+ {
+ return new Person( m_firstName, m_lastName, m_firstNameFirst );
+ }
+
+ void Person::setText( const QString& text )
+ {
+ ValueTextInterface::setText( text );
+
+ QStringList segments;
+ bool containsComma = splitName( text, segments );
+ m_firstName = "";
+ m_lastName = "";
+
+ if ( segments.isEmpty() )
+ return;
+
+ if ( !containsComma )
+ {
+ /** PubMed uses a special writing style for names, where the last name is followed by single capital letter, each being the first letter of each first name */
+ /** So, check how many single capital letters are at the end of the given segment list */
+ int singleCapitalLettersCounter = 0;
+ int p = segments.count() - 1;
+ while ( segments[p].length() == 1 && segments[p].compare( segments[p].upper() ) == 0 )
+ {
+ --p;
+ ++singleCapitalLettersCounter;
+ }
+
+ if ( singleCapitalLettersCounter > 0 )
+ {
+ /** this is a special case for names from PubMed, which are formatted like "Fischer T" */
+ /** all segment values until the first single letter segment are last name parts */
+ for ( int i = 0; i < p; ++i )
+ m_lastName.append( segments[i] ).append( " " );
+ m_lastName.append( segments[p] );
+ /** single letter segments are first name parts */
+ for ( unsigned int i = p + 1; i < segments.count() - 1; ++i )
+ m_firstName.append( segments[i] ).append( " " );
+ m_firstName.append( segments[segments.count() - 1] );
+ }
+ else
+ {
+ int from = segments.count() - 1;
+ m_lastName = segments[from];
+ /** check for lower case parts of the last name such as "van", "von", "de", ... */
+ while ( from > 0 )
+ {
+ if ( segments[from - 1].compare( segments[from - 1].lower() ) != 0 )
+ break;
+ --from;
+ m_lastName.prepend( " " );
+ m_lastName.prepend( segments[from] );
+ }
+
+ if ( from > 0 )
+ {
+ /** there are segments left for the first name */
+ m_firstName = *segments.begin();
+ for ( QStringList::Iterator it = ++segments.begin(); from > 1; ++it, --from )
+ {
+ m_firstName.append( " " );
+ m_firstName.append( *it );
+ }
+ }
+ }
+ }
+ else
+ {
+ bool inLastName = TRUE;
+ for ( unsigned int i = 0; i < segments.count();++i )
+ {
+ if ( segments[i] == "," )
+ inLastName = FALSE;
+ else if ( inLastName )
+ {
+ if ( !m_lastName.isEmpty() ) m_lastName.append( " " );
+ m_lastName.append( segments[i] );
+ }
+ else
+ {
+ if ( !m_firstName.isEmpty() ) m_firstName.append( " " );
+ m_firstName.append( segments[i] );
+ }
+ }
+ }
+ }
+
+ QString Person::text() const
+ {
+ return text( m_firstNameFirst );
+ }
+
+ QString Person::text( bool firstNameFirst ) const
+ {
+
+ if ( m_firstName.isEmpty() )
+ return m_lastName;
+ else
+ return firstNameFirst ? m_firstName + " " + m_lastName : m_lastName + ", " + m_firstName;
+ }
+
+ QString Person::firstName()
+ {
+ return m_firstName;
+ }
+ QString Person::lastName()
+ {
+ return m_lastName;
+ }
+
+ /** Splits a name into single words. If the name's text was reversed (Last, First), the result will be true and the comma will be added to segments. Otherwise the functions result will be false. This function respects protecting {...}. */
+ bool Person::splitName( const QString& text, QStringList& segments )
+ {
+ int bracketCounter = 0;
+ bool result = FALSE;
+ QString buffer = "";
+
+ for ( unsigned int pos = 0; pos < text.length(); ++pos )
+ {
+ if ( text[pos] == '{' )
+ ++bracketCounter;
+ else if ( text[pos] == '}' )
+ --bracketCounter;
+
+ if ( text[pos] == ' ' && bracketCounter == 0 )
+ {
+ if ( !buffer.isEmpty() )
+ {
+ segments.append( buffer );
+ buffer = "";
+ }
+ }
+ else if ( text[pos] == ',' && bracketCounter == 0 )
+ {
+ if ( !buffer.isEmpty() )
+ {
+ segments.append( buffer );
+ buffer = "";
+ }
+ segments.append( "," );
+ result = TRUE;
+ }
+ else
+ buffer.append( text[pos] );
+ }
+
+ if ( !buffer.isEmpty() )
+ segments.append( buffer );
+
+ return result;
+ }
+
+ PersonContainer::PersonContainer( bool firstNameFirst ) : ValueItem( QString::null ), m_firstNameFirst( firstNameFirst )
+ {
+ // nothing
+ }
+
+ PersonContainer::PersonContainer( const QString& text, bool firstNameFirst ) : ValueItem( text ), m_firstNameFirst( firstNameFirst )
+ {
+ persons.append( new Person( text, m_firstNameFirst ) );
+ }
+
+ ValueItem *PersonContainer::clone()
+ {
+ PersonContainer *result = new PersonContainer( m_firstNameFirst );
+ for ( QValueList<Person*>::ConstIterator it = persons.begin(); it != persons.end(); ++it )
+ result->persons.append(( *it )->clone() );
+
+ return result;
+ }
+
+ void PersonContainer::setText( const QString& text )
+ {
+ ValueTextInterface::setText( text );
+ qDebug( "You cannot set a text ('%s') to a Value object", text.latin1() );
+ }
+
+ QString PersonContainer::text() const
+ {
+ QString result;
+ bool first = TRUE;
+
+ for ( QValueList<Person*>::ConstIterator it = persons.begin(); it != persons.end(); ++it )
+ {
+ if ( !first )
+ result.append( " and " );
+ else
+ first = FALSE;
+ result.append(( *it )->text() );
+ }
+
+ return result;
+ }
+
+ void PersonContainer::replace( const QString &before, const QString &after )
+ {
+ for ( QValueList<Person*>::ConstIterator it = persons.begin(); it != persons.end(); ++it )
+ ( *it )->replace( before, after );
+ }
+
+ MacroKey::MacroKey( const QString& text ) : ValueItem( text )
+ {
+ m_isValid = isValidInternal();
+ }
+
+ ValueItem *MacroKey::clone()
+ {
+ return new MacroKey( text() );
+ }
+
+ void MacroKey::setText( const QString& text )
+ {
+ ValueItem::setText( text );
+ m_isValid = isValidInternal();
+ }
+
+ bool MacroKey::isValid()
+ {
+ return m_isValid;
+ }
+
+ bool MacroKey::isValidInternal()
+ {
+ return !text().contains( QRegExp( "![-.:/+_a-zA-Z0-9]" ) );
+ }
+
+ PlainText::PlainText( const QString& text ) : ValueItem( text )
+ {
+ // nothing
+ }
+
+ ValueItem *PlainText::clone()
+ {
+ return new PlainText( text() );
+ }
+
+ Value::Value() : ValueTextInterface( QString::null )
+ {
+ // nothing
+ }
+
+ Value::Value( const Value *other ) : ValueTextInterface( other )
+ {
+ for ( QValueList<ValueItem*>::ConstIterator it = other->items.begin(); it != other->items.end(); ++it )
+ items.append(( *it )->clone() );
+ }
+
+ Value::Value( const QString& text, bool isMacroKey ): ValueTextInterface( text )
+ {
+ ValueItem *item = NULL;
+ if ( isMacroKey )
+ item = new MacroKey( text );
+ else
+ item = new PlainText( text );
+ items.append( item );
+ }
+
+ void Value::setText( const QString& text )
+ {
+ ValueTextInterface::setText( text );
+ qDebug( "You cannot set a text ('%s') to a Value object", text.latin1() );
+ }
+
+ QString Value::text() const
+ {
+ QString result;
+
+ for ( QValueList<ValueItem*>::ConstIterator it = items.begin(); it != items.end(); ++it )
+ result.append(( *it )->text() );
+
+ return result;
+ }
+
+ void Value::replace( const QString &before, const QString &after )
+ {
+ for ( QValueList<ValueItem*>::ConstIterator it = items.begin(); it != items.end(); ++it )
+ ( *it )->replace( before, after );
+ }
+}