// kmsearchpattern.h // Author: Marc Mutz <Marc@Mutz.com> // This code is under GPL! #ifndef _kmsearchpattern_h_ #define _kmsearchpattern_h_ #include <tdelocale.h> #include <tqptrlist.h> #include <tqstring.h> #include <tqcstring.h> #include "kmmsgbase.h" // for KMMsgStatus class KMMessage; class TDEConfig; class DwBoyerMoore; class DwString; // maximum number of filter rules per filter const int FILTER_MAX_RULES=8; /** Incoming mail is sent through the list of mail filter rules before it is placed in the associated mail folder (usually "inbox"). This class represents one mail filter rule. It is also used to represent a search rule as used by the search dialog and folders. @short This class represents one search pattern rule. */ class KMSearchRule { public: /** Operators for comparison of field and contents. If you change the order or contents of the enum: do not forget to change funcConfigNames[], sFilterFuncList and matches() in KMSearchRule, too. Also, it is assumed that these functions come in pairs of logical opposites (ie. "=" <-> "!=", ">" <-> "<=", etc.). */ enum Function { FuncNone = -1, FuncContains=0, FuncContainsNot, FuncEquals, FuncNotEqual, FuncRegExp, FuncNotRegExp, FuncIsGreater, FuncIsLessOrEqual, FuncIsLess, FuncIsGreaterOrEqual, FuncIsInAddressbook, FuncIsNotInAddressbook, FuncIsInCategory, FuncIsNotInCategory, FuncHasAttachment, FuncHasNoAttachment}; KMSearchRule ( const TQCString & field=0, Function=FuncContains, const TQString &contents=TQString() ); KMSearchRule ( const KMSearchRule &other ); const KMSearchRule & operator=( const KMSearchRule & other ); /** Create a search rule of a certain type by instantiating the appro- priate subclass depending on the @p field. */ static KMSearchRule* createInstance( const TQCString & field=0, Function function=FuncContains, const TQString & contents=TQString() ); static KMSearchRule* createInstance( const TQCString & field, const char * function, const TQString & contents ); static KMSearchRule * createInstance( const KMSearchRule & other ); /** Initialize the object from a given config file. The group must be preset. @p aIdx is an identifier that is used to distinguish rules within a single config group. This function does no validation of the data obtained from the config file. You should call isEmpty yourself if you need valid rules. */ static KMSearchRule* createInstanceFromConfig( const TDEConfig * config, int aIdx ); virtual ~KMSearchRule() {}; /** Tries to match the rule against the given KMMessage. @return TRUE if the rule matched, FALSE otherwise. Must be implemented by subclasses. */ virtual bool matches( const KMMessage * msg ) const = 0; /** Optimized version tries to match the rule against the given @see DwString. @return TRUE if the rule matched, FALSE otherwise. */ virtual bool matches( const DwString & str, KMMessage & msg, const DwBoyerMoore * headerField=0, int headerLen=-1 ) const; /** Determine whether the rule is worth considering. It isn't if either the field is not set or the contents is empty. KFilter should make sure that it's rule list contains only non-empty rules, as matches doesn't check this. */ virtual bool isEmpty() const = 0; /** Returns true if the rule depends on a complete message, otherwise returns false. */ virtual bool requiresBody() const { return true; } /** Save the object into a given config file. The group must be preset. @p aIdx is an identifier that is used to distinguish rules within a single config group. This function will happily write itself even when it's not valid, assuming higher layers to Do The Right Thing(TM). */ void writeConfig( TDEConfig * config, int aIdx ) const; /** Return filter function. This can be any of the operators defined in Function. */ Function function() const { return mFunction; } /** Set filter function. */ void setFunction( Function aFunction ) { mFunction = aFunction; } /** Return message header field name (without the trailing ':'). There are also six pseudo-headers: @li \<message\>: Try to match against the whole message. @li \<body\>: Try to match against the body of the message. @li \<any header\>: Try to match against any header field. @li \<recipients\>: Try to match against both To: and Cc: header fields. @li \<size\>: Try to match against size of message (numerical). @li \<age in days\>: Try to match against age of message (numerical). @li \<status\>: Try to match against status of message (status). */ TQCString field() const { return mField; } /** Set message header field name (make sure there's no trailing colon ':') */ void setField( const TQCString & field ) { mField = field; } /** Return the value. This can be either a substring to search for in or a regexp pattern to match against the header. */ TQString contents() const { return mContents; } /** Set the value. */ void setContents( const TQString & aContents ) { mContents = aContents; } /** Returns the rule as string. For debugging.*/ const TQString asString() const; private: static Function configValueToFunc( const char * str ); static TQString functionToString( Function function ); TQCString mField; Function mFunction; TQString mContents; }; // subclasses representing the different kinds of searches /** This class represents a search to be performed against a string. * The string can be either a message header, or a pseudo header, such * as \<body\> @short This class represents a search pattern rule operating on a string. */ class KMSearchRuleString : public KMSearchRule { public: KMSearchRuleString( const TQCString & field=0, Function function=FuncContains, const TQString & contents=TQString() ); KMSearchRuleString( const KMSearchRuleString & other ); const KMSearchRuleString & operator=( const KMSearchRuleString & other ); virtual ~KMSearchRuleString(); virtual bool isEmpty() const ; virtual bool requiresBody() const; virtual bool matches( const KMMessage * msg ) const; /** Optimized version tries to match the rule against the given DwString. @return TRUE if the rule matched, FALSE otherwise. */ virtual bool matches( const DwString & str, KMMessage & msg, const DwBoyerMoore * headerField=0, int headerLen=-1 ) const; /** Helper for the main matches() method. Does the actual comparing. */ bool matchesInternal( const TQString & msgContents ) const; private: const DwBoyerMoore *mBmHeaderField; }; /** This class represents a search to be performed against a numerical value, * such as the age of the message in days or its size. @short This class represents a search pattern rule operating on numerical values. */ class KMSearchRuleNumerical : public KMSearchRule { public: KMSearchRuleNumerical( const TQCString & field=0, Function function=FuncContains, const TQString & contents=TQString() ); virtual bool isEmpty() const ; virtual bool matches( const KMMessage * msg ) const; /** Helper for the main matches() method. Does the actual comparing. */ bool matchesInternal( long numericalValue, long numericalMsgContents, const TQString & msgContents ) const; }; namespace KMail { // The below are used in several places and here so they are accessible. struct MessageStatus { const char * const text; const char * const icon; }; // If you change the ordering here; also do it in the enum below static const MessageStatus StatusValues[] = { { I18N_NOOP( "Important" ), "kmmsgflag" }, { I18N_NOOP( "New" ), "kmmsgnew" }, { I18N_NOOP( "Unread" ), "kmmsgunseen" }, { I18N_NOOP( "Read" ), "kmmsgread" }, { I18N_NOOP( "Old" ), 0 }, { I18N_NOOP( "Deleted" ), "kmmsgdel" }, { I18N_NOOP( "Replied" ), "kmmsgreplied" }, { I18N_NOOP( "Forwarded" ), "kmmsgforwarded" }, { I18N_NOOP( "Queued" ), "kmmsgqueued" }, { I18N_NOOP( "Sent" ), "kmmsgsent" }, { I18N_NOOP( "Watched" ), "kmmsgwatched" }, { I18N_NOOP( "Ignored" ), "kmmsgignored" }, { I18N_NOOP( "Spam" ), "kmmsgspam" }, { I18N_NOOP( "Ham" ), "kmmsgham" }, { I18N_NOOP( "To Do" ), "kmmsgtodo" }, { I18N_NOOP( "Invitation" ), "kmmsginvitation" }, { I18N_NOOP( "Has Attachment"), "kmmsgattachment" } //must be last }; // If you change the ordering here; also do it in the array above enum StatusValueTypes { StatusImportant = 0, StatusNew = 1, StatusUnread = 2, StatusRead = 3, StatusOld = 4, StatusDeleted = 5, StatusReplied = 6, StatusForwarded = 7, StatusQueued = 8, StatusSent = 9, StatusWatched = 10, StatusIgnored = 11, StatusSpam = 12, StatusHam = 13, StatusToDo = 14, StatusInvitation = 15, StatusHasAttachment = 16 //must be last }; static const int StatusValueCount = sizeof( StatusValues ) / sizeof( MessageStatus ); // we want to show all status entries in the quick search bar, but only the // ones up to attachment in the search/filter dialog, because there the // attachment case is handled separately. static const int StatusValueCountWithoutHidden = StatusValueCount - 1; } /** This class represents a search to be performed against the status of a * messsage. The status is represented by a bitfield. @short This class represents a search pattern rule operating on message status. */ class KMSearchRuleStatus : public KMSearchRule { public: KMSearchRuleStatus( const TQCString & field=0, Function function=FuncContains, const TQString & contents=TQString() ); KMSearchRuleStatus( int status, Function function=FuncContains ); virtual bool isEmpty() const ; virtual bool matches( const KMMessage * msg ) const; //Not possible to implement this form for status searching virtual bool matches( const DwString &, KMMessage &, const DwBoyerMoore *, int ) const; static KMMsgStatus statusFromEnglishName(const TQString&); private: KMMsgStatus mStatus; }; // ------------------------------------------------------------------------ /** This class is an abstraction of a search over messages. It is intended to be used inside a KFilter (which adds KFilterAction's), as well as in KMSearch. It can read and write itself into a TDEConfig group and there is a constructor, mainly used by KMFilter to initialize from a preset TDEConfig-Group. From a class hierarchy point of view, it is a TQPtrList of KMSearchRule's that adds the boolean operators (see Operator) 'and' and 'or' that connect the rules logically, and has a name under which it could be stored in the config file. As a TQPtrList with autoDelete enabled, it assumes that it is the central repository for the rules it contains. So if you want to reuse a rule in another pattern, make a deep copy of that rule. @short An abstraction of a search over messages. @author Marc Mutz <Marc@Mutz.com> */ class KMSearchPattern : public TQPtrList<KMSearchRule> { public: /** Boolean operators that connect the return values of the individual rules. A pattern with @p OpAnd will match iff all it's rules match, whereas a pattern with @p OpOr will match iff any of it's rules matches. */ enum Operator { OpAnd, OpOr }; /** Constructor that initializes from a given TDEConfig group, if given. This feature is mainly (solely?) used in KMFilter, as we don't allow to store search patterns in the config (yet). If config is 0, provides a pattern with minimal, but sufficient initialization. Unmodified, such a pattern will fail to match any KMMessage. You can query for such an empty rule by using isEmpty, which is inherited from TQPtrList. */ KMSearchPattern( const TDEConfig * config=0 ); /** Destructor. Deletes all stored rules! */ ~KMSearchPattern(); /** The central function of this class. Tries to match the set of rules against a KMMessage. It's virtual to allow derived classes with added rules to reimplement it, yet reimplemented methods should and (&&) the result of this function with their own result or else most functionality is lacking, or has to be reimplemented, since the rules are private to this class. @return TRUE if the match was successful, FALSE otherwise. */ bool matches( const KMMessage * msg, bool ignoreBody = false ) const; bool matches( const DwString & str, bool ignoreBody = false ) const; bool matches( TQ_UINT32 sernum, bool ignoreBody = false ) const; /** Returns true if the pattern only depends the DwString that backs a message */ bool requiresBody() const; /** Removes all empty rules from the list. You should call this method whenever the user had had control of the rules outside of this class. (e.g. after editing it with KMSearchPatternEdit). */ void purify(); /** Reads a search pattern from a TDEConfig. The group has to be preset. If it does not find a valid saerch pattern in the preset group, initializes the pattern as if it were constructed using the default constructor. For backwards compatibility with previous versions of KMail, it checks for old-style filter rules (e.g. using @p OpIgnore) in @p config und converts them to the new format on writeConfig. Derived classes reimplementing readConfig() should also call this method, or else the rules will not be loaded. */ void readConfig( const TDEConfig * config ); /** Writes itself into @p config. The group has to be preset. Tries to delete old-style keys by overwriting them with TQString(). Derived classes reimplementing writeConfig() should also call this method, or else the rules will not be stored. */ void writeConfig( TDEConfig * config ) const; /** Get the name of the search pattern. */ TQString name() const { return mName; } /** Set the name of the search pattern. KMFilter uses this to store it's own name, too. */ void setName( const TQString & newName ) { mName = newName ; } /** Get the filter operator */ KMSearchPattern::Operator op() const { return mOperator; } /** Set the filter operator */ void setOp( KMSearchPattern::Operator aOp ) { mOperator = aOp; } /** Returns the pattern as string. For debugging.*/ TQString asString() const; /** Overloaded assignment operator. Makes a deep copy. */ const KMSearchPattern & operator=( const KMSearchPattern & aPattern ); private: /** Tries to import a legacy search pattern, ie. one that still has e.g. the @p unless or @p ignore operator which were useful as long as the number of rules was restricted to two. This method is called from readConfig, which detects legacy configurations and also makes sure that this method is called from an initialized object. */ void importLegacyConfig( const TDEConfig * config ); /** Initializes the object. Clears the list of rules, sets the name to "<i18n("unnamed")>", and the boolean operator to @p OpAnd. */ void init(); TQString mName; Operator mOperator; }; #endif /* _kmsearchpattern_h_ */