-rw-r--r-- | libopie/pim/obackendfactory.h | 20 | ||||
-rw-r--r-- | libopie/pim/opimaccessbackend.h | 40 | ||||
-rw-r--r-- | libopie/pim/opimaccesstemplate.h | 57 | ||||
-rw-r--r-- | libopie/pim/opimcache.h | 117 | ||||
-rw-r--r-- | libopie/pim/orecordlist.h | 20 | ||||
-rw-r--r-- | libopie/pim/otemplatebase.h | 11 | ||||
-rw-r--r-- | libopie2/opiepim/backend/obackendfactory.h | 20 | ||||
-rw-r--r-- | libopie2/opiepim/backend/opimaccessbackend.h | 40 | ||||
-rw-r--r-- | libopie2/opiepim/core/opimaccesstemplate.h | 57 | ||||
-rw-r--r-- | libopie2/opiepim/core/opimcache.h | 117 | ||||
-rw-r--r-- | libopie2/opiepim/core/otemplatebase.h | 11 | ||||
-rw-r--r-- | libopie2/opiepim/orecordlist.h | 20 |
12 files changed, 496 insertions, 34 deletions
diff --git a/libopie/pim/obackendfactory.h b/libopie/pim/obackendfactory.h index f11f029..89b8c58 100644 --- a/libopie/pim/obackendfactory.h +++ b/libopie/pim/obackendfactory.h | |||
@@ -3,95 +3,103 @@ | |||
3 | * | 3 | * |
4 | * Copyright (c) 2002 by Stefan Eilers (Eilers.Stefan@epost.de) | 4 | * Copyright (c) 2002 by Stefan Eilers (Eilers.Stefan@epost.de) |
5 | * | 5 | * |
6 | * ===================================================================== | 6 | * ===================================================================== |
7 | *This program is free software; you can redistribute it and/or | 7 | *This program is free software; you can redistribute it and/or |
8 | *modify it under the terms of the GNU Library General Public | 8 | *modify it under the terms of the GNU Library General Public |
9 | * License as published by the Free Software Foundation; | 9 | * License as published by the Free Software Foundation; |
10 | * either version 2 of the License, or (at your option) any later | 10 | * either version 2 of the License, or (at your option) any later |
11 | * version. | 11 | * version. |
12 | * ===================================================================== | 12 | * ===================================================================== |
13 | * ToDo: Use plugins | 13 | * ToDo: Use plugins |
14 | * ===================================================================== | 14 | * ===================================================================== |
15 | * Version: $Id$ | 15 | * Version: $Id$ |
16 | * ===================================================================== | 16 | * ===================================================================== |
17 | * History: | 17 | * History: |
18 | * $Log$ | 18 | * $Log$ |
19 | * Revision 1.3 2002/10/10 17:08:58 zecke | ||
20 | * The Cache is finally in place | ||
21 | * I tested it with my todolist and it 'works' for 10.000 todos the hits are awesome ;) | ||
22 | * The read ahead functionality does not make sense for XMLs backends because most of the stuff is already in memory. While using readahead on SQL makes things a lot faster.... | ||
23 | * I still have to fully implement read ahead | ||
24 | * This change is bic but sc | ||
25 | * | ||
19 | * Revision 1.2 2002/10/08 09:27:36 eilers | 26 | * Revision 1.2 2002/10/08 09:27:36 eilers |
20 | * Fixed libopie.pro to include the new pim-API. | 27 | * Fixed libopie.pro to include the new pim-API. |
21 | * The SQL-Stuff is currently deactivated. Otherwise everyone who wants to | 28 | * The SQL-Stuff is currently deactivated. Otherwise everyone who wants to |
22 | * compile itself would need to install libsqlite, libopiesql... | 29 | * compile itself would need to install libsqlite, libopiesql... |
23 | * Therefore, the backend currently uses XML only.. | 30 | * Therefore, the backend currently uses XML only.. |
24 | * | 31 | * |
25 | * Revision 1.1 2002/10/07 17:35:01 eilers | 32 | * Revision 1.1 2002/10/07 17:35:01 eilers |
26 | * added OBackendFactory for advanced backend access | 33 | * added OBackendFactory for advanced backend access |
27 | * | 34 | * |
28 | * | 35 | * |
29 | * ===================================================================== | 36 | * ===================================================================== |
30 | */ | 37 | */ |
31 | #ifndef __OPIE_BACKENDFACTORY_H_ | 38 | #ifndef OPIE_BACKENDFACTORY_H_ |
32 | #define __OPIE_BACKENDFACTORY_H_ | 39 | #define OPIE_BACKENDFACTORY_H_ |
33 | 40 | ||
34 | #include <qstring.h> | 41 | #include <qstring.h> |
35 | #include <qasciidict.h> | 42 | #include <qasciidict.h> |
36 | #include <qpe/config.h> | 43 | #include <qpe/config.h> |
37 | 44 | ||
38 | #include "otodoaccessxml.h" | 45 | #include "otodoaccessxml.h" |
39 | #include "ocontactaccessbackend_xml.h" | 46 | #include "ocontactaccessbackend_xml.h" |
40 | 47 | ||
41 | #ifdef __USE_SQL | 48 | /*#ifdef __USE_SQL |
42 | #include "otodoaccesssql.h" | 49 | #include "otodoaccesssql.h" |
43 | #endif | 50 | #endif |
51 | */ | ||
44 | 52 | ||
45 | template<class T> | 53 | template<class T> |
46 | class OBackendFactory | 54 | class OBackendFactory |
47 | { | 55 | { |
48 | public: | 56 | public: |
49 | OBackendFactory() {}; | 57 | OBackendFactory() {}; |
50 | 58 | ||
51 | enum BACKENDS { | 59 | enum BACKENDS { |
52 | TODO, | 60 | TODO, |
53 | CONTACT, | 61 | CONTACT, |
54 | DATE | 62 | DATE |
55 | }; | 63 | }; |
56 | 64 | ||
57 | static T* Default( const QString backendName, const QString& appName ){ | 65 | static T* Default( const QString backendName, const QString& appName ){ |
58 | 66 | ||
59 | 67 | ||
60 | Config config( "pimaccess" ); | 68 | Config config( "pimaccess" ); |
61 | config.setGroup ( backendName ); | 69 | config.setGroup ( backendName ); |
62 | QString backend = config.readEntry( "usebackend" ); | 70 | QString backend = config.readEntry( "usebackend" ); |
63 | 71 | ||
64 | QAsciiDict<int> dict ( 3 ); | 72 | QAsciiDict<int> dict ( 3 ); |
65 | dict.setAutoDelete ( TRUE ); | 73 | dict.setAutoDelete ( TRUE ); |
66 | 74 | ||
67 | dict.insert( "todo", new int (TODO) ); | 75 | dict.insert( "todo", new int (TODO) ); |
68 | dict.insert( "contact", new int (CONTACT) ); | 76 | dict.insert( "contact", new int (CONTACT) ); |
69 | 77 | ||
70 | qWarning ("TODO is: %d", TODO); | 78 | qWarning ("TODO is: %d", TODO); |
71 | qWarning ("CONTACT is: %d", CONTACT); | 79 | qWarning ("CONTACT is: %d", CONTACT); |
72 | 80 | ||
73 | switch ( *dict.take( backendName ) ){ | 81 | switch ( *dict.take( backendName ) ){ |
74 | case TODO: | 82 | case TODO: |
75 | #ifdef __USE_SQL | 83 | /*#ifdef __USE_SQL |
76 | if ( backend == "sql" ) | 84 | if ( backend == "sql" ) |
77 | return (T*) new OTodoAccessBackendSQL(""); | 85 | return (T*) new OTodoAccessBackendSQL(""); |
78 | #else | 86 | #else*/ |
79 | if ( backend == "sql" ) | 87 | if ( backend == "sql" ) |
80 | qWarning ("OBackendFactory:: sql Backend not implemented! Using XML instead!"); | 88 | qWarning ("OBackendFactory:: sql Backend not implemented! Using XML instead!"); |
81 | #endif | 89 | //#endif |
82 | 90 | ||
83 | return (T*) new OTodoAccessXML( appName ); | 91 | return (T*) new OTodoAccessXML( appName ); |
84 | case CONTACT: | 92 | case CONTACT: |
85 | if ( backend == "sql" ) | 93 | if ( backend == "sql" ) |
86 | qWarning ("OBackendFactory:: sql Backend not implemented! Using XML instead!"); | 94 | qWarning ("OBackendFactory:: sql Backend not implemented! Using XML instead!"); |
87 | 95 | ||
88 | return (T*) new OContactAccessBackend_XML( appName ); | 96 | return (T*) new OContactAccessBackend_XML( appName ); |
89 | case DATE: | 97 | case DATE: |
90 | qWarning ("OBackendFactory:: DATE-Backend not implemented!"); | 98 | qWarning ("OBackendFactory:: DATE-Backend not implemented!"); |
91 | return NULL; | 99 | return NULL; |
92 | default: | 100 | default: |
93 | return NULL; | 101 | return NULL; |
94 | } | 102 | } |
95 | 103 | ||
96 | 104 | ||
97 | } | 105 | } |
diff --git a/libopie/pim/opimaccessbackend.h b/libopie/pim/opimaccessbackend.h index c27acbb..27d3cb8 100644 --- a/libopie/pim/opimaccessbackend.h +++ b/libopie/pim/opimaccessbackend.h | |||
@@ -1,35 +1,37 @@ | |||
1 | #ifndef OPIE_PIM_ACCESS_BACKEND | 1 | #ifndef OPIE_PIM_ACCESS_BACKEND |
2 | #define OPIE_PIM_ACCESS_BACKEND | 2 | #define OPIE_PIM_ACCESS_BACKEND |
3 | 3 | ||
4 | #include <qarray.h> | 4 | #include <qarray.h> |
5 | 5 | ||
6 | #include <opie/otemplatebase.h> | ||
6 | #include <opie/opimrecord.h> | 7 | #include <opie/opimrecord.h> |
7 | 8 | ||
8 | 9 | ||
9 | /** | 10 | /** |
10 | * OPimAccessBackend is the base class | 11 | * OPimAccessBackend is the base class |
11 | * for all private backends | 12 | * for all private backends |
12 | * it operates on OPimRecord as the base class | 13 | * it operates on OPimRecord as the base class |
13 | * and it's responsible for fast manipulating | 14 | * and it's responsible for fast manipulating |
14 | * the resource the implementation takes care | 15 | * the resource the implementation takes care |
15 | * of | 16 | * of |
16 | */ | 17 | */ |
17 | template <class T = OPimRecord> | 18 | template <class T = OPimRecord> |
18 | class OPimAccessBackend { | 19 | class OPimAccessBackend { |
19 | public: | 20 | public: |
21 | typedef OTemplateBase<T> Frontend; | ||
20 | OPimAccessBackend(); | 22 | OPimAccessBackend(); |
21 | virtual ~OPimAccessBackend(); | 23 | virtual ~OPimAccessBackend(); |
22 | 24 | ||
23 | /** | 25 | /** |
24 | * load the resource | 26 | * load the resource |
25 | */ | 27 | */ |
26 | virtual bool load() = 0; | 28 | virtual bool load() = 0; |
27 | 29 | ||
28 | /** | 30 | /** |
29 | * reload the resource | 31 | * reload the resource |
30 | */ | 32 | */ |
31 | virtual bool reload() = 0; | 33 | virtual bool reload() = 0; |
32 | 34 | ||
33 | /** | 35 | /** |
34 | * save the resource and | 36 | * save the resource and |
35 | * all it's changes | 37 | * all it's changes |
@@ -41,49 +43,85 @@ public: | |||
41 | * all available uids | 43 | * all available uids |
42 | */ | 44 | */ |
43 | virtual QArray<int> allRecords()const = 0; | 45 | virtual QArray<int> allRecords()const = 0; |
44 | 46 | ||
45 | /** | 47 | /** |
46 | * queryByExample for T with the SortOrder | 48 | * queryByExample for T with the SortOrder |
47 | * sort | 49 | * sort |
48 | */ | 50 | */ |
49 | virtual QArray<int> queryByExample( const T& t, int sort ) = 0; | 51 | virtual QArray<int> queryByExample( const T& t, int sort ) = 0; |
50 | 52 | ||
51 | /** | 53 | /** |
52 | * find the OPimRecord with uid @param uid | 54 | * find the OPimRecord with uid @param uid |
53 | * returns T and T.isEmpty() if nothing was found | 55 | * returns T and T.isEmpty() if nothing was found |
54 | */ | 56 | */ |
55 | virtual T find(int uid )const = 0; | 57 | virtual T find(int uid )const = 0; |
56 | 58 | ||
59 | virtual T find(int uid, const QArray<int>& items, | ||
60 | uint current, Frontend::CacheDirection )const ; | ||
57 | /** | 61 | /** |
58 | * clear the back end | 62 | * clear the back end |
59 | */ | 63 | */ |
60 | virtual void clear() = 0; | 64 | virtual void clear() = 0; |
61 | 65 | ||
62 | /** | 66 | /** |
63 | * add T | 67 | * add T |
64 | */ | 68 | */ |
65 | virtual bool add( const T& t ) = 0; | 69 | virtual bool add( const T& t ) = 0; |
66 | 70 | ||
67 | /** | 71 | /** |
68 | * remove | 72 | * remove |
69 | */ | 73 | */ |
70 | virtual bool remove( int uid ) = 0; | 74 | virtual bool remove( int uid ) = 0; |
71 | 75 | ||
72 | /** | 76 | /** |
73 | * replace a record with T.uid() | 77 | * replace a record with T.uid() |
74 | */ | 78 | */ |
75 | virtual bool replace( const T& t ) = 0; | 79 | virtual bool replace( const T& t ) = 0; |
76 | 80 | ||
81 | /* | ||
82 | * setTheFrontEnd!!! | ||
83 | */ | ||
84 | void setFrontend( Frontend* front ); | ||
85 | |||
86 | protected: | ||
87 | void cache( const T& t )const; | ||
88 | |||
89 | /** | ||
90 | * use a prime number here! | ||
91 | */ | ||
92 | void setSaneCacheSize( int ); | ||
93 | |||
94 | private: | ||
95 | Frontend* m_front; | ||
77 | 96 | ||
78 | }; | 97 | }; |
79 | 98 | ||
80 | template <class T> | 99 | template <class T> |
81 | OPimAccessBackend<T>::OPimAccessBackend() { | 100 | OPimAccessBackend<T>::OPimAccessBackend() { |
82 | 101 | m_front = 0l; | |
83 | } | 102 | } |
84 | template <class T> | 103 | template <class T> |
85 | OPimAccessBackend<T>::~OPimAccessBackend() { | 104 | OPimAccessBackend<T>::~OPimAccessBackend() { |
86 | 105 | ||
87 | } | 106 | } |
107 | template <class T> | ||
108 | void OPimAccessBackend<T>::setFrontend( Frontend* fr ) { | ||
109 | m_front = fr; | ||
110 | } | ||
111 | template <class T> | ||
112 | void OPimAccessBackend<T>::cache( const T& t )const { | ||
113 | if (m_front ) | ||
114 | m_front->cache( t ); | ||
115 | } | ||
116 | template <class T> | ||
117 | void OPimAccessBackend<T>::setSaneCacheSize( int size) { | ||
118 | if (m_front ) | ||
119 | m_front->setSaneCacheSize( size ); | ||
120 | } | ||
121 | template <class T> | ||
122 | T OPimAccessBackend<T>::find( int uid, const QArray<int>&, | ||
123 | uint, Frontend::CacheDirection )const { | ||
124 | return find( uid ); | ||
125 | } | ||
88 | 126 | ||
89 | #endif | 127 | #endif |
diff --git a/libopie/pim/opimaccesstemplate.h b/libopie/pim/opimaccesstemplate.h index 31ab516..92d7192 100644 --- a/libopie/pim/opimaccesstemplate.h +++ b/libopie/pim/opimaccesstemplate.h | |||
@@ -1,41 +1,43 @@ | |||
1 | #ifndef OPIE_PIM_ACCESS_TEMPLATE_H | 1 | #ifndef OPIE_PIM_ACCESS_TEMPLATE_H |
2 | #define OPIE_PIM_ACCESS_TEMPLATE_H | 2 | #define OPIE_PIM_ACCESS_TEMPLATE_H |
3 | 3 | ||
4 | #include <qarray.h> | 4 | #include <qarray.h> |
5 | 5 | ||
6 | #include <opie/opimrecord.h> | 6 | #include <opie/opimrecord.h> |
7 | #include <opie/opimaccessbackend.h> | 7 | #include <opie/opimaccessbackend.h> |
8 | #include <opie/orecordlist.h> | 8 | #include <opie/orecordlist.h> |
9 | 9 | ||
10 | #include "opimcache.h" | ||
10 | #include "otemplatebase.h" | 11 | #include "otemplatebase.h" |
11 | 12 | ||
12 | /** | 13 | /** |
13 | * Thats the frontend to our OPIE PIM | 14 | * Thats the frontend to our OPIE PIM |
14 | * Library. Either you want to use it's | 15 | * Library. Either you want to use it's |
15 | * interface or you want to implement | 16 | * interface or you want to implement |
16 | * your own Access lib | 17 | * your own Access lib |
17 | * Just create a OPimRecord and inherit from | 18 | * Just create a OPimRecord and inherit from |
18 | * the plugins | 19 | * the plugins |
19 | */ | 20 | */ |
20 | 21 | ||
21 | template <class T = OPimRecord > | 22 | template <class T = OPimRecord > |
22 | class OPimAccessTemplate : public OTemplateBase<T> { | 23 | class OPimAccessTemplate : public OTemplateBase<T> { |
23 | public: | 24 | public: |
24 | typedef ORecordList<T> List; | 25 | typedef ORecordList<T> List; |
25 | typedef OPimAccessBackend<T> BackEnd; | 26 | typedef OPimAccessBackend<T> BackEnd; |
27 | typedef OPimCache<T> Cache; | ||
26 | 28 | ||
27 | /** | 29 | /** |
28 | * our sort order | 30 | * our sort order |
29 | * should be safe explaining | 31 | * should be safe explaining |
30 | */ | 32 | */ |
31 | enum SortOrder { WildCards = 0, IgnoreCase = 1, | 33 | enum SortOrder { WildCards = 0, IgnoreCase = 1, |
32 | RegExp = 2, ExactMatch = 4 }; | 34 | RegExp = 2, ExactMatch = 4 }; |
33 | 35 | ||
34 | /** | 36 | /** |
35 | * c'tor BackEnd | 37 | * c'tor BackEnd |
36 | */ | 38 | */ |
37 | OPimAccessTemplate( BackEnd* end); | 39 | OPimAccessTemplate( BackEnd* end); |
38 | virtual ~OPimAccessTemplate(); | 40 | virtual ~OPimAccessTemplate(); |
39 | 41 | ||
40 | /** | 42 | /** |
41 | * load from the backend | 43 | * load from the backend |
@@ -60,137 +62,186 @@ public: | |||
60 | /** | 62 | /** |
61 | * return a List of records | 63 | * return a List of records |
62 | * you can iterate over them | 64 | * you can iterate over them |
63 | */ | 65 | */ |
64 | virtual List allRecords()const; | 66 | virtual List allRecords()const; |
65 | 67 | ||
66 | /** | 68 | /** |
67 | * queryByExample | 69 | * queryByExample |
68 | */ | 70 | */ |
69 | virtual List queryByExample( const T& t, int sortOrder ); | 71 | virtual List queryByExample( const T& t, int sortOrder ); |
70 | 72 | ||
71 | /** | 73 | /** |
72 | * find the OPimRecord uid | 74 | * find the OPimRecord uid |
73 | */ | 75 | */ |
74 | virtual T find( int uid )const; | 76 | virtual T find( int uid )const; |
75 | 77 | ||
78 | /** | ||
79 | * read ahead cache find method ;) | ||
80 | */ | ||
81 | virtual T find( int uid, const QArray<int>&, | ||
82 | uint current, CacheDirection dir = Forward )const; | ||
83 | |||
76 | /* invalidate cache here */ | 84 | /* invalidate cache here */ |
77 | /** | 85 | /** |
78 | * clears the backend and invalidates the backend | 86 | * clears the backend and invalidates the backend |
79 | */ | 87 | */ |
80 | virtual void clear() ; | 88 | virtual void clear() ; |
81 | 89 | ||
82 | /** | 90 | /** |
83 | * add T to the backend | 91 | * add T to the backend |
84 | */ | 92 | */ |
85 | virtual bool add( const T& t ) ; | 93 | virtual bool add( const T& t ) ; |
86 | 94 | ||
87 | /* only the uid matters */ | 95 | /* only the uid matters */ |
88 | /** | 96 | /** |
89 | * remove T from the backend | 97 | * remove T from the backend |
90 | */ | 98 | */ |
91 | virtual bool remove( const T& t ); | 99 | virtual bool remove( const T& t ); |
92 | 100 | ||
93 | /** | 101 | /** |
94 | * remove the OPimRecord with uid | 102 | * remove the OPimRecord with uid |
95 | */ | 103 | */ |
96 | virtual bool remove( int uid ); | 104 | virtual bool remove( int uid ); |
97 | 105 | ||
98 | /** | 106 | /** |
99 | * replace T from backend | 107 | * replace T from backend |
100 | */ | 108 | */ |
101 | virtual bool replace( const T& t) ; | 109 | virtual bool replace( const T& t) ; |
110 | |||
111 | /** | ||
112 | * @internal | ||
113 | */ | ||
114 | void cache( const T& )const; | ||
115 | void setSaneCacheSize( int ); | ||
102 | protected: | 116 | protected: |
103 | /** | 117 | /** |
104 | * invalidate the cache | 118 | * invalidate the cache |
105 | */ | 119 | */ |
106 | void invalidateCache(); | 120 | void invalidateCache(); |
107 | 121 | ||
108 | void setBackEnd( BackEnd* end ); | 122 | void setBackEnd( BackEnd* end ); |
109 | /** | 123 | /** |
110 | * returns the backend | 124 | * returns the backend |
111 | */ | 125 | */ |
112 | BackEnd* backEnd(); | 126 | BackEnd* backEnd(); |
113 | BackEnd* m_backEnd; | 127 | BackEnd* m_backEnd; |
128 | Cache m_cache; | ||
114 | 129 | ||
115 | }; | 130 | }; |
116 | 131 | ||
117 | template <class T> | 132 | template <class T> |
118 | OPimAccessTemplate<T>::OPimAccessTemplate( BackEnd* end ) | 133 | OPimAccessTemplate<T>::OPimAccessTemplate( BackEnd* end ) |
119 | : OTemplateBase<T>(), m_backEnd( end ) | 134 | : OTemplateBase<T>(), m_backEnd( end ) |
120 | { | 135 | { |
121 | 136 | if (end ) | |
137 | end->setFrontend( this ); | ||
122 | } | 138 | } |
123 | template <class T> | 139 | template <class T> |
124 | OPimAccessTemplate<T>::~OPimAccessTemplate() { | 140 | OPimAccessTemplate<T>::~OPimAccessTemplate() { |
125 | qWarning("~OPimAccessTemplate<T>"); | 141 | qWarning("~OPimAccessTemplate<T>"); |
126 | delete m_backEnd; | 142 | delete m_backEnd; |
127 | } | 143 | } |
128 | template <class T> | 144 | template <class T> |
129 | bool OPimAccessTemplate<T>::load() { | 145 | bool OPimAccessTemplate<T>::load() { |
146 | invalidateCache(); | ||
130 | return m_backEnd->load(); | 147 | return m_backEnd->load(); |
131 | } | 148 | } |
132 | template <class T> | 149 | template <class T> |
133 | bool OPimAccessTemplate<T>::reload() { | 150 | bool OPimAccessTemplate<T>::reload() { |
134 | return m_backEnd->reload(); | 151 | return m_backEnd->reload(); |
135 | } | 152 | } |
136 | template <class T> | 153 | template <class T> |
137 | bool OPimAccessTemplate<T>::save() { | 154 | bool OPimAccessTemplate<T>::save() { |
138 | return m_backEnd->save(); | 155 | return m_backEnd->save(); |
139 | } | 156 | } |
140 | template <class T> | 157 | template <class T> |
141 | OPimAccessTemplate<T>::List OPimAccessTemplate<T>::allRecords()const { | 158 | OPimAccessTemplate<T>::List OPimAccessTemplate<T>::allRecords()const { |
142 | QArray<int> ints = m_backEnd->allRecords(); | 159 | QArray<int> ints = m_backEnd->allRecords(); |
143 | List lis(ints, this ); | 160 | List lis(ints, this ); |
144 | return lis; | 161 | return lis; |
145 | } | 162 | } |
146 | template <class T> | 163 | template <class T> |
147 | OPimAccessTemplate<T>::List | 164 | OPimAccessTemplate<T>::List |
148 | OPimAccessTemplate<T>::queryByExample( const T& t, int sortOrder ) { | 165 | OPimAccessTemplate<T>::queryByExample( const T& t, int sortOrder ) { |
149 | QArray<int> ints = m_backEnd->queryByExample( t, sortOrder ); | 166 | QArray<int> ints = m_backEnd->queryByExample( t, sortOrder ); |
150 | 167 | ||
151 | List lis(ints, this ); | 168 | List lis(ints, this ); |
152 | return lis; | 169 | return lis; |
153 | } | 170 | } |
154 | template <class T> | 171 | template <class T> |
155 | T OPimAccessTemplate<T>::find( int uid ) const{ | 172 | T OPimAccessTemplate<T>::find( int uid ) const{ |
156 | T t = m_backEnd->find( uid ); | 173 | T t = m_backEnd->find( uid ); |
174 | cache( t ); | ||
175 | return t; | ||
176 | } | ||
177 | template <class T> | ||
178 | T OPimAccessTemplate<T>::find( int uid, const QArray<int>& ar, | ||
179 | uint current, CacheDirection dir )const { | ||
180 | /* | ||
181 | * better do T.isEmpty() | ||
182 | * after a find this way we would | ||
183 | * avoid two finds in QCache... | ||
184 | */ | ||
185 | // qWarning("find it now %d", uid ); | ||
186 | if (m_cache.contains( uid ) ) { | ||
187 | qWarning("m cache contains %d", uid); | ||
188 | return m_cache.find( uid ); | ||
189 | } | ||
190 | |||
191 | T t = m_backEnd->find( uid, ar, current, dir ); | ||
192 | qWarning("found it and cache it now %d", uid); | ||
193 | cache( t ); | ||
157 | return t; | 194 | return t; |
158 | } | 195 | } |
159 | template <class T> | 196 | template <class T> |
160 | void OPimAccessTemplate<T>::clear() { | 197 | void OPimAccessTemplate<T>::clear() { |
161 | invalidateCache(); | 198 | invalidateCache(); |
162 | m_backEnd->clear(); | 199 | m_backEnd->clear(); |
163 | } | 200 | } |
164 | template <class T> | 201 | template <class T> |
165 | bool OPimAccessTemplate<T>::add( const T& t ) { | 202 | bool OPimAccessTemplate<T>::add( const T& t ) { |
203 | cache( t ); | ||
166 | return m_backEnd->add( t ); | 204 | return m_backEnd->add( t ); |
167 | } | 205 | } |
168 | template <class T> | 206 | template <class T> |
169 | bool OPimAccessTemplate<T>::remove( const T& t ) { | 207 | bool OPimAccessTemplate<T>::remove( const T& t ) { |
170 | return m_backEnd->remove( t.uid() ); | 208 | return remove( t.uid() ); |
171 | } | 209 | } |
172 | template <class T> | 210 | template <class T> |
173 | bool OPimAccessTemplate<T>::remove( int uid ) { | 211 | bool OPimAccessTemplate<T>::remove( int uid ) { |
212 | m_cache.remove( uid ); | ||
174 | return m_backEnd->remove( uid ); | 213 | return m_backEnd->remove( uid ); |
175 | } | 214 | } |
176 | template <class T> | 215 | template <class T> |
177 | bool OPimAccessTemplate<T>::replace( const T& t ) { | 216 | bool OPimAccessTemplate<T>::replace( const T& t ) { |
217 | m_cache.replace( t ); | ||
178 | return m_backEnd->replace( t ); | 218 | return m_backEnd->replace( t ); |
179 | } | 219 | } |
180 | template <class T> | 220 | template <class T> |
181 | void OPimAccessTemplate<T>::invalidateCache() { | 221 | void OPimAccessTemplate<T>::invalidateCache() { |
182 | 222 | m_cache.invalidate(); | |
183 | } | 223 | } |
184 | template <class T> | 224 | template <class T> |
185 | OPimAccessTemplate<T>::BackEnd* OPimAccessTemplate<T>::backEnd() { | 225 | OPimAccessTemplate<T>::BackEnd* OPimAccessTemplate<T>::backEnd() { |
186 | return m_backEnd; | 226 | return m_backEnd; |
187 | } | 227 | } |
188 | template <class T> | 228 | template <class T> |
189 | bool OPimAccessTemplate<T>::wasChangedExternally()const { | 229 | bool OPimAccessTemplate<T>::wasChangedExternally()const { |
190 | return false; | 230 | return false; |
191 | } | 231 | } |
192 | template <class T> | 232 | template <class T> |
193 | void OPimAccessTemplate<T>::setBackEnd( BackEnd* end ) { | 233 | void OPimAccessTemplate<T>::setBackEnd( BackEnd* end ) { |
194 | m_backEnd = end; | 234 | m_backEnd = end; |
235 | if (m_backEnd ) | ||
236 | m_backEnd->setFrontend( this ); | ||
237 | } | ||
238 | template <class T> | ||
239 | void OPimAccessTemplate<T>::cache( const T& t ) const{ | ||
240 | /* hacky we need to work around the const*/ | ||
241 | ((OPimAccessTemplate<T>*)this)->m_cache.add( t ); | ||
242 | } | ||
243 | template <class T> | ||
244 | void OPimAccessTemplate<T>::setSaneCacheSize( int size ) { | ||
245 | m_cache.setSize( size ); | ||
195 | } | 246 | } |
196 | #endif | 247 | #endif |
diff --git a/libopie/pim/opimcache.h b/libopie/pim/opimcache.h new file mode 100644 index 0000000..067f6e7 --- a/dev/null +++ b/libopie/pim/opimcache.h | |||
@@ -0,0 +1,117 @@ | |||
1 | #ifndef OPIE_PIM_CACHE_H | ||
2 | #define OPIE_PIM_CACHE_H | ||
3 | |||
4 | #include <qintcache.h> | ||
5 | |||
6 | #include "opimrecord.h" | ||
7 | |||
8 | template <class T = OPimRecord> | ||
9 | class OPimCacheItem { | ||
10 | public: | ||
11 | OPimCacheItem( const T& t = T() ); | ||
12 | ~OPimCacheItem(); | ||
13 | |||
14 | T record()const; | ||
15 | void setRecord( const T& ); | ||
16 | private: | ||
17 | T m_t; | ||
18 | }; | ||
19 | |||
20 | /** | ||
21 | * OPimCache for caching the items | ||
22 | * We support adding, removing | ||
23 | * and finding | ||
24 | */ | ||
25 | template <class T = OPimRecord> | ||
26 | class OPimCache { | ||
27 | public: | ||
28 | typedef OPimCacheItem<T> Item; | ||
29 | OPimCache(); | ||
30 | ~OPimCache(); | ||
31 | |||
32 | bool contains(int uid)const; | ||
33 | void invalidate(); | ||
34 | void setSize( int size ); | ||
35 | |||
36 | T find(int uid )const; | ||
37 | void add( const T& ); | ||
38 | void remove( int uid ); | ||
39 | void replace( const T& ); | ||
40 | |||
41 | private: | ||
42 | QIntCache<Item> m_cache; | ||
43 | }; | ||
44 | |||
45 | // Implementation | ||
46 | template <class T> | ||
47 | OPimCacheItem<T>::OPimCacheItem( const T& t ) | ||
48 | : m_t(t) { | ||
49 | } | ||
50 | template <class T> | ||
51 | OPimCacheItem<T>::~OPimCacheItem() { | ||
52 | |||
53 | } | ||
54 | template <class T> | ||
55 | T OPimCacheItem<T>::record()const { | ||
56 | return m_t; | ||
57 | } | ||
58 | template <class T> | ||
59 | void OPimCacheItem<T>::setRecord( const T& t ) { | ||
60 | m_t = t; | ||
61 | } | ||
62 | // Cache | ||
63 | template <class T> | ||
64 | OPimCache<T>::OPimCache() { | ||
65 | m_cache.setAutoDelete( TRUE ); | ||
66 | } | ||
67 | template <class T> | ||
68 | OPimCache<T>::~OPimCache() { | ||
69 | |||
70 | } | ||
71 | template <class T> | ||
72 | bool OPimCache<T>::contains(int uid )const { | ||
73 | Item* it = m_cache.find( uid, FALSE ); | ||
74 | if (!it) | ||
75 | return false; | ||
76 | return true; | ||
77 | } | ||
78 | template <class T> | ||
79 | void OPimCache<T>::invalidate() { | ||
80 | m_cache.clear(); | ||
81 | } | ||
82 | template <class T> | ||
83 | void OPimCache<T>::setSize( int size ) { | ||
84 | m_cache.setMaxCost( size ); | ||
85 | } | ||
86 | template <class T> | ||
87 | T OPimCache<T>::find(int uid )const { | ||
88 | Item *it = m_cache.find( uid ); | ||
89 | if (it) | ||
90 | return it->record(); | ||
91 | return T(); | ||
92 | } | ||
93 | template <class T> | ||
94 | void OPimCache<T>::add( const T& t ) { | ||
95 | Item* it = 0l; | ||
96 | it = m_cache.find(t.uid(), FALSE ); | ||
97 | |||
98 | if (it ) | ||
99 | it->setRecord( t ); | ||
100 | |||
101 | it = new Item( t ); | ||
102 | if (!m_cache.insert( t.uid(), it ) ) | ||
103 | delete it; | ||
104 | } | ||
105 | template <class T> | ||
106 | void OPimCache<T>::remove( int uid ) { | ||
107 | m_cache.remove( uid ); | ||
108 | } | ||
109 | template <class T> | ||
110 | void OPimCache<T>::replace( const T& t) { | ||
111 | Item *it = m_cache.find( t.uid() ); | ||
112 | if ( it ) { | ||
113 | it->setRecord( t ); | ||
114 | } | ||
115 | } | ||
116 | |||
117 | #endif | ||
diff --git a/libopie/pim/orecordlist.h b/libopie/pim/orecordlist.h index b6fa7fa..08f5c85 100644 --- a/libopie/pim/orecordlist.h +++ b/libopie/pim/orecordlist.h | |||
@@ -55,32 +55,33 @@ public: | |||
55 | /** | 55 | /** |
56 | * the number of items | 56 | * the number of items |
57 | */ | 57 | */ |
58 | uint count()const; | 58 | uint count()const; |
59 | 59 | ||
60 | /** | 60 | /** |
61 | * sets the current item | 61 | * sets the current item |
62 | */ | 62 | */ |
63 | void setCurrent( uint cur ); | 63 | void setCurrent( uint cur ); |
64 | 64 | ||
65 | private: | 65 | private: |
66 | QArray<int> m_uids; | 66 | QArray<int> m_uids; |
67 | uint m_current; | 67 | uint m_current; |
68 | const Base* m_temp; | 68 | const Base* m_temp; |
69 | bool m_end : 1; | 69 | bool m_end : 1; |
70 | T m_record; | 70 | T m_record; |
71 | bool m_direction :1; | ||
71 | 72 | ||
72 | /* d pointer for future versions */ | 73 | /* d pointer for future versions */ |
73 | class IteratorPrivate; | 74 | class IteratorPrivate; |
74 | IteratorPrivate *d; | 75 | IteratorPrivate *d; |
75 | }; | 76 | }; |
76 | /** | 77 | /** |
77 | * The recordlist used as a return type | 78 | * The recordlist used as a return type |
78 | * from OPimAccessTemplate | 79 | * from OPimAccessTemplate |
79 | */ | 80 | */ |
80 | template <class T = OPimRecord > | 81 | template <class T = OPimRecord > |
81 | class ORecordList { | 82 | class ORecordList { |
82 | public: | 83 | public: |
83 | typedef OTemplateBase<T> Base; | 84 | typedef OTemplateBase<T> Base; |
84 | typedef ORecordListIterator<T> Iterator; | 85 | typedef ORecordListIterator<T> Iterator; |
85 | 86 | ||
86 | /** | 87 | /** |
@@ -112,115 +113,119 @@ public: | |||
112 | /* | 113 | /* |
113 | ConstIterator begin()const; | 114 | ConstIterator begin()const; |
114 | ConstIterator end()const; | 115 | ConstIterator end()const; |
115 | */ | 116 | */ |
116 | private: | 117 | private: |
117 | QArray<int> m_ids; | 118 | QArray<int> m_ids; |
118 | const Base* m_acc; | 119 | const Base* m_acc; |
119 | }; | 120 | }; |
120 | 121 | ||
121 | /* ok now implement it */ | 122 | /* ok now implement it */ |
122 | template <class T> | 123 | template <class T> |
123 | ORecordListIterator<T>::ORecordListIterator() { | 124 | ORecordListIterator<T>::ORecordListIterator() { |
124 | m_current = 0; | 125 | m_current = 0; |
125 | m_temp = 0l; | 126 | m_temp = 0l; |
126 | m_end = true; | 127 | m_end = true; |
127 | m_record = T(); | 128 | m_record = T(); |
129 | /* forward */ | ||
130 | m_direction = TRUE; | ||
128 | } | 131 | } |
129 | template <class T> | 132 | template <class T> |
130 | ORecordListIterator<T>::~ORecordListIterator() { | 133 | ORecordListIterator<T>::~ORecordListIterator() { |
131 | /* nothing to delete */ | 134 | /* nothing to delete */ |
132 | } | 135 | } |
133 | 136 | ||
134 | template <class T> | 137 | template <class T> |
135 | ORecordListIterator<T>::ORecordListIterator( const ORecordListIterator<T>& it) { | 138 | ORecordListIterator<T>::ORecordListIterator( const ORecordListIterator<T>& it) { |
136 | // qWarning("ORecordListIterator copy c'tor"); | 139 | // qWarning("ORecordListIterator copy c'tor"); |
137 | m_uids = it.m_uids; | 140 | m_uids = it.m_uids; |
138 | m_current = it.m_current; | 141 | m_current = it.m_current; |
139 | m_temp = it.m_temp; | 142 | m_temp = it.m_temp; |
140 | m_end = it.m_end; | 143 | m_end = it.m_end; |
141 | m_record = it.m_record; | 144 | m_record = it.m_record; |
145 | m_direction = it.m_direction; | ||
142 | } | 146 | } |
143 | 147 | ||
144 | template <class T> | 148 | template <class T> |
145 | ORecordListIterator<T> &ORecordListIterator<T>::operator=( const ORecordListIterator<T>& it) { | 149 | ORecordListIterator<T> &ORecordListIterator<T>::operator=( const ORecordListIterator<T>& it) { |
146 | m_uids = it.m_uids; | 150 | m_uids = it.m_uids; |
147 | m_current = it.m_current; | 151 | m_current = it.m_current; |
148 | m_temp = it.m_temp; | 152 | m_temp = it.m_temp; |
149 | m_end = it.m_end; | 153 | m_end = it.m_end; |
150 | m_record = it.m_record; | 154 | m_record = it.m_record; |
151 | 155 | ||
152 | return *this; | 156 | return *this; |
153 | } | 157 | } |
154 | 158 | ||
155 | template <class T> | 159 | template <class T> |
156 | T ORecordListIterator<T>::operator*() { | 160 | T ORecordListIterator<T>::operator*() { |
157 | qWarning("operator* %d %d", m_current, m_uids[m_current] ); | 161 | qWarning("operator* %d %d", m_current, m_uids[m_current] ); |
158 | if (!m_end ) | 162 | if (!m_end ) |
159 | /* FIXME | 163 | m_record = m_temp->find( m_uids[m_current], m_uids, m_current, |
160 | * until the cache is in place | 164 | m_direction ? Base::Forward : |
161 | * we do the uid match uid check | 165 | Base::Reverse ); |
162 | */ | ||
163 | m_record = m_temp->find( m_uids[m_current] ); | ||
164 | else | 166 | else |
165 | m_record = T(); | 167 | m_record = T(); |
166 | 168 | ||
167 | return m_record; | 169 | return m_record; |
168 | } | 170 | } |
169 | 171 | ||
170 | template <class T> | 172 | template <class T> |
171 | ORecordListIterator<T> &ORecordListIterator<T>::operator++() { | 173 | ORecordListIterator<T> &ORecordListIterator<T>::operator++() { |
174 | m_direction = true; | ||
172 | if (m_current < m_uids.count() ) { | 175 | if (m_current < m_uids.count() ) { |
173 | m_end = false; | 176 | m_end = false; |
174 | ++m_current; | 177 | ++m_current; |
175 | }else | 178 | }else |
176 | m_end = true; | 179 | m_end = true; |
177 | 180 | ||
178 | return *this; | 181 | return *this; |
179 | } | 182 | } |
180 | template <class T> | 183 | template <class T> |
181 | ORecordListIterator<T> &ORecordListIterator<T>::operator--() { | 184 | ORecordListIterator<T> &ORecordListIterator<T>::operator--() { |
185 | m_direction = false; | ||
182 | if ( m_current > 0 ) { | 186 | if ( m_current > 0 ) { |
183 | --m_current; | 187 | --m_current; |
184 | m_end = false; | 188 | m_end = false; |
185 | } else | 189 | } else |
186 | m_end = true; | 190 | m_end = true; |
187 | 191 | ||
188 | return *this; | 192 | return *this; |
189 | } | 193 | } |
190 | 194 | ||
191 | template <class T> | 195 | template <class T> |
192 | bool ORecordListIterator<T>::operator==( const ORecordListIterator<T>& it ) { | 196 | bool ORecordListIterator<T>::operator==( const ORecordListIterator<T>& it ) { |
193 | 197 | ||
194 | /* if both are at we're the same.... */ | 198 | /* if both are at we're the same.... */ |
195 | if ( m_end == it.m_end ) return true; | 199 | if ( m_end == it.m_end ) return true; |
196 | 200 | ||
197 | if ( m_uids != it.m_uids ) return false; | 201 | if ( m_uids != it.m_uids ) return false; |
198 | if ( m_current != it.m_current ) return false; | 202 | if ( m_current != it.m_current ) return false; |
199 | if ( m_temp != it.m_temp ) return false; | 203 | if ( m_temp != it.m_temp ) return false; |
200 | 204 | ||
201 | return true; | 205 | return true; |
202 | } | 206 | } |
203 | template <class T> | 207 | template <class T> |
204 | bool ORecordListIterator<T>::operator!=( const ORecordListIterator<T>& it ) { | 208 | bool ORecordListIterator<T>::operator!=( const ORecordListIterator<T>& it ) { |
205 | return !(*this == it ); | 209 | return !(*this == it ); |
206 | } | 210 | } |
207 | template <class T> | 211 | template <class T> |
208 | ORecordListIterator<T>::ORecordListIterator( const QArray<int> uids, | 212 | ORecordListIterator<T>::ORecordListIterator( const QArray<int> uids, |
209 | const Base* t ) | 213 | const Base* t ) |
210 | : m_uids( uids ), m_current( 0 ), m_temp( t ), m_end( false ) | 214 | : m_uids( uids ), m_current( 0 ), m_temp( t ), m_end( false ), |
215 | m_direction( false ) | ||
211 | { | 216 | { |
212 | } | 217 | } |
213 | template <class T> | 218 | template <class T> |
214 | uint ORecordListIterator<T>::current()const { | 219 | uint ORecordListIterator<T>::current()const { |
215 | return m_current; | 220 | return m_current; |
216 | } | 221 | } |
217 | template <class T> | 222 | template <class T> |
218 | void ORecordListIterator<T>::setCurrent( uint cur ) { | 223 | void ORecordListIterator<T>::setCurrent( uint cur ) { |
219 | if( cur < m_uids.count() ) { | 224 | if( cur < m_uids.count() ) { |
220 | m_end = false; | 225 | m_end = false; |
221 | m_current= cur; | 226 | m_current= cur; |
222 | } | 227 | } |
223 | } | 228 | } |
224 | template <class T> | 229 | template <class T> |
225 | uint ORecordListIterator<T>::count()const { | 230 | uint ORecordListIterator<T>::count()const { |
226 | return m_uids.count(); | 231 | return m_uids.count(); |
@@ -241,19 +246,20 @@ ORecordList<T>::Iterator ORecordList<T>::begin() { | |||
241 | return it; | 246 | return it; |
242 | } | 247 | } |
243 | template <class T> | 248 | template <class T> |
244 | ORecordList<T>::Iterator ORecordList<T>::end() { | 249 | ORecordList<T>::Iterator ORecordList<T>::end() { |
245 | Iterator it( m_ids, m_acc ); | 250 | Iterator it( m_ids, m_acc ); |
246 | it.m_end = true; | 251 | it.m_end = true; |
247 | it.m_current = m_ids.count(); | 252 | it.m_current = m_ids.count(); |
248 | 253 | ||
249 | return it; | 254 | return it; |
250 | } | 255 | } |
251 | template <class T> | 256 | template <class T> |
252 | uint ORecordList<T>::count()const { | 257 | uint ORecordList<T>::count()const { |
253 | return m_ids.count(); | 258 | return m_ids.count(); |
254 | } | 259 | } |
255 | template <class T> | 260 | template <class T> |
256 | T ORecordList<T>::operator[]( uint i ) { | 261 | T ORecordList<T>::operator[]( uint i ) { |
257 | return m_acc->find( m_ids[i] ); | 262 | /* forward */ |
263 | return m_acc->find( m_ids[i], m_ids, i ); | ||
258 | } | 264 | } |
259 | #endif | 265 | #endif |
diff --git a/libopie/pim/otemplatebase.h b/libopie/pim/otemplatebase.h index f71417b..b855919 100644 --- a/libopie/pim/otemplatebase.h +++ b/libopie/pim/otemplatebase.h | |||
@@ -1,21 +1,32 @@ | |||
1 | #ifndef OPIE_TEMPLATE_BASE_H | 1 | #ifndef OPIE_TEMPLATE_BASE_H |
2 | #define OPIE_TEMPLATE_BASE_H | 2 | #define OPIE_TEMPLATE_BASE_H |
3 | 3 | ||
4 | #include <qarray.h> | ||
5 | |||
4 | #include "opimrecord.h" | 6 | #include "opimrecord.h" |
5 | 7 | ||
6 | /** | 8 | /** |
7 | * internal template base | 9 | * internal template base |
8 | */ | 10 | */ |
9 | template <class T = OPimRecord> | 11 | template <class T = OPimRecord> |
10 | class OTemplateBase { | 12 | class OTemplateBase { |
11 | public: | 13 | public: |
14 | enum CacheDirection { Forward=0, Reverse }; | ||
12 | OTemplateBase() { | 15 | OTemplateBase() { |
13 | }; | 16 | }; |
14 | virtual ~OTemplateBase() { | 17 | virtual ~OTemplateBase() { |
15 | } | 18 | } |
16 | virtual T find( int uid )const = 0; | 19 | virtual T find( int uid )const = 0; |
17 | 20 | ||
21 | /** | ||
22 | * read ahead find | ||
23 | */ | ||
24 | virtual T find( int uid, const QArray<int>& items, | ||
25 | uint current, CacheDirection dir = Forward )const = 0; | ||
26 | virtual void cache( const T& )const = 0; | ||
27 | virtual void setSaneCacheSize( int ) = 0; | ||
28 | |||
18 | }; | 29 | }; |
19 | 30 | ||
20 | 31 | ||
21 | #endif | 32 | #endif |
diff --git a/libopie2/opiepim/backend/obackendfactory.h b/libopie2/opiepim/backend/obackendfactory.h index f11f029..89b8c58 100644 --- a/libopie2/opiepim/backend/obackendfactory.h +++ b/libopie2/opiepim/backend/obackendfactory.h | |||
@@ -3,95 +3,103 @@ | |||
3 | * | 3 | * |
4 | * Copyright (c) 2002 by Stefan Eilers (Eilers.Stefan@epost.de) | 4 | * Copyright (c) 2002 by Stefan Eilers (Eilers.Stefan@epost.de) |
5 | * | 5 | * |
6 | * ===================================================================== | 6 | * ===================================================================== |
7 | *This program is free software; you can redistribute it and/or | 7 | *This program is free software; you can redistribute it and/or |
8 | *modify it under the terms of the GNU Library General Public | 8 | *modify it under the terms of the GNU Library General Public |
9 | * License as published by the Free Software Foundation; | 9 | * License as published by the Free Software Foundation; |
10 | * either version 2 of the License, or (at your option) any later | 10 | * either version 2 of the License, or (at your option) any later |
11 | * version. | 11 | * version. |
12 | * ===================================================================== | 12 | * ===================================================================== |
13 | * ToDo: Use plugins | 13 | * ToDo: Use plugins |
14 | * ===================================================================== | 14 | * ===================================================================== |
15 | * Version: $Id$ | 15 | * Version: $Id$ |
16 | * ===================================================================== | 16 | * ===================================================================== |
17 | * History: | 17 | * History: |
18 | * $Log$ | 18 | * $Log$ |
19 | * Revision 1.3 2002/10/10 17:08:58 zecke | ||
20 | * The Cache is finally in place | ||
21 | * I tested it with my todolist and it 'works' for 10.000 todos the hits are awesome ;) | ||
22 | * The read ahead functionality does not make sense for XMLs backends because most of the stuff is already in memory. While using readahead on SQL makes things a lot faster.... | ||
23 | * I still have to fully implement read ahead | ||
24 | * This change is bic but sc | ||
25 | * | ||
19 | * Revision 1.2 2002/10/08 09:27:36 eilers | 26 | * Revision 1.2 2002/10/08 09:27:36 eilers |
20 | * Fixed libopie.pro to include the new pim-API. | 27 | * Fixed libopie.pro to include the new pim-API. |
21 | * The SQL-Stuff is currently deactivated. Otherwise everyone who wants to | 28 | * The SQL-Stuff is currently deactivated. Otherwise everyone who wants to |
22 | * compile itself would need to install libsqlite, libopiesql... | 29 | * compile itself would need to install libsqlite, libopiesql... |
23 | * Therefore, the backend currently uses XML only.. | 30 | * Therefore, the backend currently uses XML only.. |
24 | * | 31 | * |
25 | * Revision 1.1 2002/10/07 17:35:01 eilers | 32 | * Revision 1.1 2002/10/07 17:35:01 eilers |
26 | * added OBackendFactory for advanced backend access | 33 | * added OBackendFactory for advanced backend access |
27 | * | 34 | * |
28 | * | 35 | * |
29 | * ===================================================================== | 36 | * ===================================================================== |
30 | */ | 37 | */ |
31 | #ifndef __OPIE_BACKENDFACTORY_H_ | 38 | #ifndef OPIE_BACKENDFACTORY_H_ |
32 | #define __OPIE_BACKENDFACTORY_H_ | 39 | #define OPIE_BACKENDFACTORY_H_ |
33 | 40 | ||
34 | #include <qstring.h> | 41 | #include <qstring.h> |
35 | #include <qasciidict.h> | 42 | #include <qasciidict.h> |
36 | #include <qpe/config.h> | 43 | #include <qpe/config.h> |
37 | 44 | ||
38 | #include "otodoaccessxml.h" | 45 | #include "otodoaccessxml.h" |
39 | #include "ocontactaccessbackend_xml.h" | 46 | #include "ocontactaccessbackend_xml.h" |
40 | 47 | ||
41 | #ifdef __USE_SQL | 48 | /*#ifdef __USE_SQL |
42 | #include "otodoaccesssql.h" | 49 | #include "otodoaccesssql.h" |
43 | #endif | 50 | #endif |
51 | */ | ||
44 | 52 | ||
45 | template<class T> | 53 | template<class T> |
46 | class OBackendFactory | 54 | class OBackendFactory |
47 | { | 55 | { |
48 | public: | 56 | public: |
49 | OBackendFactory() {}; | 57 | OBackendFactory() {}; |
50 | 58 | ||
51 | enum BACKENDS { | 59 | enum BACKENDS { |
52 | TODO, | 60 | TODO, |
53 | CONTACT, | 61 | CONTACT, |
54 | DATE | 62 | DATE |
55 | }; | 63 | }; |
56 | 64 | ||
57 | static T* Default( const QString backendName, const QString& appName ){ | 65 | static T* Default( const QString backendName, const QString& appName ){ |
58 | 66 | ||
59 | 67 | ||
60 | Config config( "pimaccess" ); | 68 | Config config( "pimaccess" ); |
61 | config.setGroup ( backendName ); | 69 | config.setGroup ( backendName ); |
62 | QString backend = config.readEntry( "usebackend" ); | 70 | QString backend = config.readEntry( "usebackend" ); |
63 | 71 | ||
64 | QAsciiDict<int> dict ( 3 ); | 72 | QAsciiDict<int> dict ( 3 ); |
65 | dict.setAutoDelete ( TRUE ); | 73 | dict.setAutoDelete ( TRUE ); |
66 | 74 | ||
67 | dict.insert( "todo", new int (TODO) ); | 75 | dict.insert( "todo", new int (TODO) ); |
68 | dict.insert( "contact", new int (CONTACT) ); | 76 | dict.insert( "contact", new int (CONTACT) ); |
69 | 77 | ||
70 | qWarning ("TODO is: %d", TODO); | 78 | qWarning ("TODO is: %d", TODO); |
71 | qWarning ("CONTACT is: %d", CONTACT); | 79 | qWarning ("CONTACT is: %d", CONTACT); |
72 | 80 | ||
73 | switch ( *dict.take( backendName ) ){ | 81 | switch ( *dict.take( backendName ) ){ |
74 | case TODO: | 82 | case TODO: |
75 | #ifdef __USE_SQL | 83 | /*#ifdef __USE_SQL |
76 | if ( backend == "sql" ) | 84 | if ( backend == "sql" ) |
77 | return (T*) new OTodoAccessBackendSQL(""); | 85 | return (T*) new OTodoAccessBackendSQL(""); |
78 | #else | 86 | #else*/ |
79 | if ( backend == "sql" ) | 87 | if ( backend == "sql" ) |
80 | qWarning ("OBackendFactory:: sql Backend not implemented! Using XML instead!"); | 88 | qWarning ("OBackendFactory:: sql Backend not implemented! Using XML instead!"); |
81 | #endif | 89 | //#endif |
82 | 90 | ||
83 | return (T*) new OTodoAccessXML( appName ); | 91 | return (T*) new OTodoAccessXML( appName ); |
84 | case CONTACT: | 92 | case CONTACT: |
85 | if ( backend == "sql" ) | 93 | if ( backend == "sql" ) |
86 | qWarning ("OBackendFactory:: sql Backend not implemented! Using XML instead!"); | 94 | qWarning ("OBackendFactory:: sql Backend not implemented! Using XML instead!"); |
87 | 95 | ||
88 | return (T*) new OContactAccessBackend_XML( appName ); | 96 | return (T*) new OContactAccessBackend_XML( appName ); |
89 | case DATE: | 97 | case DATE: |
90 | qWarning ("OBackendFactory:: DATE-Backend not implemented!"); | 98 | qWarning ("OBackendFactory:: DATE-Backend not implemented!"); |
91 | return NULL; | 99 | return NULL; |
92 | default: | 100 | default: |
93 | return NULL; | 101 | return NULL; |
94 | } | 102 | } |
95 | 103 | ||
96 | 104 | ||
97 | } | 105 | } |
diff --git a/libopie2/opiepim/backend/opimaccessbackend.h b/libopie2/opiepim/backend/opimaccessbackend.h index c27acbb..27d3cb8 100644 --- a/libopie2/opiepim/backend/opimaccessbackend.h +++ b/libopie2/opiepim/backend/opimaccessbackend.h | |||
@@ -1,35 +1,37 @@ | |||
1 | #ifndef OPIE_PIM_ACCESS_BACKEND | 1 | #ifndef OPIE_PIM_ACCESS_BACKEND |
2 | #define OPIE_PIM_ACCESS_BACKEND | 2 | #define OPIE_PIM_ACCESS_BACKEND |
3 | 3 | ||
4 | #include <qarray.h> | 4 | #include <qarray.h> |
5 | 5 | ||
6 | #include <opie/otemplatebase.h> | ||
6 | #include <opie/opimrecord.h> | 7 | #include <opie/opimrecord.h> |
7 | 8 | ||
8 | 9 | ||
9 | /** | 10 | /** |
10 | * OPimAccessBackend is the base class | 11 | * OPimAccessBackend is the base class |
11 | * for all private backends | 12 | * for all private backends |
12 | * it operates on OPimRecord as the base class | 13 | * it operates on OPimRecord as the base class |
13 | * and it's responsible for fast manipulating | 14 | * and it's responsible for fast manipulating |
14 | * the resource the implementation takes care | 15 | * the resource the implementation takes care |
15 | * of | 16 | * of |
16 | */ | 17 | */ |
17 | template <class T = OPimRecord> | 18 | template <class T = OPimRecord> |
18 | class OPimAccessBackend { | 19 | class OPimAccessBackend { |
19 | public: | 20 | public: |
21 | typedef OTemplateBase<T> Frontend; | ||
20 | OPimAccessBackend(); | 22 | OPimAccessBackend(); |
21 | virtual ~OPimAccessBackend(); | 23 | virtual ~OPimAccessBackend(); |
22 | 24 | ||
23 | /** | 25 | /** |
24 | * load the resource | 26 | * load the resource |
25 | */ | 27 | */ |
26 | virtual bool load() = 0; | 28 | virtual bool load() = 0; |
27 | 29 | ||
28 | /** | 30 | /** |
29 | * reload the resource | 31 | * reload the resource |
30 | */ | 32 | */ |
31 | virtual bool reload() = 0; | 33 | virtual bool reload() = 0; |
32 | 34 | ||
33 | /** | 35 | /** |
34 | * save the resource and | 36 | * save the resource and |
35 | * all it's changes | 37 | * all it's changes |
@@ -41,49 +43,85 @@ public: | |||
41 | * all available uids | 43 | * all available uids |
42 | */ | 44 | */ |
43 | virtual QArray<int> allRecords()const = 0; | 45 | virtual QArray<int> allRecords()const = 0; |
44 | 46 | ||
45 | /** | 47 | /** |
46 | * queryByExample for T with the SortOrder | 48 | * queryByExample for T with the SortOrder |
47 | * sort | 49 | * sort |
48 | */ | 50 | */ |
49 | virtual QArray<int> queryByExample( const T& t, int sort ) = 0; | 51 | virtual QArray<int> queryByExample( const T& t, int sort ) = 0; |
50 | 52 | ||
51 | /** | 53 | /** |
52 | * find the OPimRecord with uid @param uid | 54 | * find the OPimRecord with uid @param uid |
53 | * returns T and T.isEmpty() if nothing was found | 55 | * returns T and T.isEmpty() if nothing was found |
54 | */ | 56 | */ |
55 | virtual T find(int uid )const = 0; | 57 | virtual T find(int uid )const = 0; |
56 | 58 | ||
59 | virtual T find(int uid, const QArray<int>& items, | ||
60 | uint current, Frontend::CacheDirection )const ; | ||
57 | /** | 61 | /** |
58 | * clear the back end | 62 | * clear the back end |
59 | */ | 63 | */ |
60 | virtual void clear() = 0; | 64 | virtual void clear() = 0; |
61 | 65 | ||
62 | /** | 66 | /** |
63 | * add T | 67 | * add T |
64 | */ | 68 | */ |
65 | virtual bool add( const T& t ) = 0; | 69 | virtual bool add( const T& t ) = 0; |
66 | 70 | ||
67 | /** | 71 | /** |
68 | * remove | 72 | * remove |
69 | */ | 73 | */ |
70 | virtual bool remove( int uid ) = 0; | 74 | virtual bool remove( int uid ) = 0; |
71 | 75 | ||
72 | /** | 76 | /** |
73 | * replace a record with T.uid() | 77 | * replace a record with T.uid() |
74 | */ | 78 | */ |
75 | virtual bool replace( const T& t ) = 0; | 79 | virtual bool replace( const T& t ) = 0; |
76 | 80 | ||
81 | /* | ||
82 | * setTheFrontEnd!!! | ||
83 | */ | ||
84 | void setFrontend( Frontend* front ); | ||
85 | |||
86 | protected: | ||
87 | void cache( const T& t )const; | ||
88 | |||
89 | /** | ||
90 | * use a prime number here! | ||
91 | */ | ||
92 | void setSaneCacheSize( int ); | ||
93 | |||
94 | private: | ||
95 | Frontend* m_front; | ||
77 | 96 | ||
78 | }; | 97 | }; |
79 | 98 | ||
80 | template <class T> | 99 | template <class T> |
81 | OPimAccessBackend<T>::OPimAccessBackend() { | 100 | OPimAccessBackend<T>::OPimAccessBackend() { |
82 | 101 | m_front = 0l; | |
83 | } | 102 | } |
84 | template <class T> | 103 | template <class T> |
85 | OPimAccessBackend<T>::~OPimAccessBackend() { | 104 | OPimAccessBackend<T>::~OPimAccessBackend() { |
86 | 105 | ||
87 | } | 106 | } |
107 | template <class T> | ||
108 | void OPimAccessBackend<T>::setFrontend( Frontend* fr ) { | ||
109 | m_front = fr; | ||
110 | } | ||
111 | template <class T> | ||
112 | void OPimAccessBackend<T>::cache( const T& t )const { | ||
113 | if (m_front ) | ||
114 | m_front->cache( t ); | ||
115 | } | ||
116 | template <class T> | ||
117 | void OPimAccessBackend<T>::setSaneCacheSize( int size) { | ||
118 | if (m_front ) | ||
119 | m_front->setSaneCacheSize( size ); | ||
120 | } | ||
121 | template <class T> | ||
122 | T OPimAccessBackend<T>::find( int uid, const QArray<int>&, | ||
123 | uint, Frontend::CacheDirection )const { | ||
124 | return find( uid ); | ||
125 | } | ||
88 | 126 | ||
89 | #endif | 127 | #endif |
diff --git a/libopie2/opiepim/core/opimaccesstemplate.h b/libopie2/opiepim/core/opimaccesstemplate.h index 31ab516..92d7192 100644 --- a/libopie2/opiepim/core/opimaccesstemplate.h +++ b/libopie2/opiepim/core/opimaccesstemplate.h | |||
@@ -1,41 +1,43 @@ | |||
1 | #ifndef OPIE_PIM_ACCESS_TEMPLATE_H | 1 | #ifndef OPIE_PIM_ACCESS_TEMPLATE_H |
2 | #define OPIE_PIM_ACCESS_TEMPLATE_H | 2 | #define OPIE_PIM_ACCESS_TEMPLATE_H |
3 | 3 | ||
4 | #include <qarray.h> | 4 | #include <qarray.h> |
5 | 5 | ||
6 | #include <opie/opimrecord.h> | 6 | #include <opie/opimrecord.h> |
7 | #include <opie/opimaccessbackend.h> | 7 | #include <opie/opimaccessbackend.h> |
8 | #include <opie/orecordlist.h> | 8 | #include <opie/orecordlist.h> |
9 | 9 | ||
10 | #include "opimcache.h" | ||
10 | #include "otemplatebase.h" | 11 | #include "otemplatebase.h" |
11 | 12 | ||
12 | /** | 13 | /** |
13 | * Thats the frontend to our OPIE PIM | 14 | * Thats the frontend to our OPIE PIM |
14 | * Library. Either you want to use it's | 15 | * Library. Either you want to use it's |
15 | * interface or you want to implement | 16 | * interface or you want to implement |
16 | * your own Access lib | 17 | * your own Access lib |
17 | * Just create a OPimRecord and inherit from | 18 | * Just create a OPimRecord and inherit from |
18 | * the plugins | 19 | * the plugins |
19 | */ | 20 | */ |
20 | 21 | ||
21 | template <class T = OPimRecord > | 22 | template <class T = OPimRecord > |
22 | class OPimAccessTemplate : public OTemplateBase<T> { | 23 | class OPimAccessTemplate : public OTemplateBase<T> { |
23 | public: | 24 | public: |
24 | typedef ORecordList<T> List; | 25 | typedef ORecordList<T> List; |
25 | typedef OPimAccessBackend<T> BackEnd; | 26 | typedef OPimAccessBackend<T> BackEnd; |
27 | typedef OPimCache<T> Cache; | ||
26 | 28 | ||
27 | /** | 29 | /** |
28 | * our sort order | 30 | * our sort order |
29 | * should be safe explaining | 31 | * should be safe explaining |
30 | */ | 32 | */ |
31 | enum SortOrder { WildCards = 0, IgnoreCase = 1, | 33 | enum SortOrder { WildCards = 0, IgnoreCase = 1, |
32 | RegExp = 2, ExactMatch = 4 }; | 34 | RegExp = 2, ExactMatch = 4 }; |
33 | 35 | ||
34 | /** | 36 | /** |
35 | * c'tor BackEnd | 37 | * c'tor BackEnd |
36 | */ | 38 | */ |
37 | OPimAccessTemplate( BackEnd* end); | 39 | OPimAccessTemplate( BackEnd* end); |
38 | virtual ~OPimAccessTemplate(); | 40 | virtual ~OPimAccessTemplate(); |
39 | 41 | ||
40 | /** | 42 | /** |
41 | * load from the backend | 43 | * load from the backend |
@@ -60,137 +62,186 @@ public: | |||
60 | /** | 62 | /** |
61 | * return a List of records | 63 | * return a List of records |
62 | * you can iterate over them | 64 | * you can iterate over them |
63 | */ | 65 | */ |
64 | virtual List allRecords()const; | 66 | virtual List allRecords()const; |
65 | 67 | ||
66 | /** | 68 | /** |
67 | * queryByExample | 69 | * queryByExample |
68 | */ | 70 | */ |
69 | virtual List queryByExample( const T& t, int sortOrder ); | 71 | virtual List queryByExample( const T& t, int sortOrder ); |
70 | 72 | ||
71 | /** | 73 | /** |
72 | * find the OPimRecord uid | 74 | * find the OPimRecord uid |
73 | */ | 75 | */ |
74 | virtual T find( int uid )const; | 76 | virtual T find( int uid )const; |
75 | 77 | ||
78 | /** | ||
79 | * read ahead cache find method ;) | ||
80 | */ | ||
81 | virtual T find( int uid, const QArray<int>&, | ||
82 | uint current, CacheDirection dir = Forward )const; | ||
83 | |||
76 | /* invalidate cache here */ | 84 | /* invalidate cache here */ |
77 | /** | 85 | /** |
78 | * clears the backend and invalidates the backend | 86 | * clears the backend and invalidates the backend |
79 | */ | 87 | */ |
80 | virtual void clear() ; | 88 | virtual void clear() ; |
81 | 89 | ||
82 | /** | 90 | /** |
83 | * add T to the backend | 91 | * add T to the backend |
84 | */ | 92 | */ |
85 | virtual bool add( const T& t ) ; | 93 | virtual bool add( const T& t ) ; |
86 | 94 | ||
87 | /* only the uid matters */ | 95 | /* only the uid matters */ |
88 | /** | 96 | /** |
89 | * remove T from the backend | 97 | * remove T from the backend |
90 | */ | 98 | */ |
91 | virtual bool remove( const T& t ); | 99 | virtual bool remove( const T& t ); |
92 | 100 | ||
93 | /** | 101 | /** |
94 | * remove the OPimRecord with uid | 102 | * remove the OPimRecord with uid |
95 | */ | 103 | */ |
96 | virtual bool remove( int uid ); | 104 | virtual bool remove( int uid ); |
97 | 105 | ||
98 | /** | 106 | /** |
99 | * replace T from backend | 107 | * replace T from backend |
100 | */ | 108 | */ |
101 | virtual bool replace( const T& t) ; | 109 | virtual bool replace( const T& t) ; |
110 | |||
111 | /** | ||
112 | * @internal | ||
113 | */ | ||
114 | void cache( const T& )const; | ||
115 | void setSaneCacheSize( int ); | ||
102 | protected: | 116 | protected: |
103 | /** | 117 | /** |
104 | * invalidate the cache | 118 | * invalidate the cache |
105 | */ | 119 | */ |
106 | void invalidateCache(); | 120 | void invalidateCache(); |
107 | 121 | ||
108 | void setBackEnd( BackEnd* end ); | 122 | void setBackEnd( BackEnd* end ); |
109 | /** | 123 | /** |
110 | * returns the backend | 124 | * returns the backend |
111 | */ | 125 | */ |
112 | BackEnd* backEnd(); | 126 | BackEnd* backEnd(); |
113 | BackEnd* m_backEnd; | 127 | BackEnd* m_backEnd; |
128 | Cache m_cache; | ||
114 | 129 | ||
115 | }; | 130 | }; |
116 | 131 | ||
117 | template <class T> | 132 | template <class T> |
118 | OPimAccessTemplate<T>::OPimAccessTemplate( BackEnd* end ) | 133 | OPimAccessTemplate<T>::OPimAccessTemplate( BackEnd* end ) |
119 | : OTemplateBase<T>(), m_backEnd( end ) | 134 | : OTemplateBase<T>(), m_backEnd( end ) |
120 | { | 135 | { |
121 | 136 | if (end ) | |
137 | end->setFrontend( this ); | ||
122 | } | 138 | } |
123 | template <class T> | 139 | template <class T> |
124 | OPimAccessTemplate<T>::~OPimAccessTemplate() { | 140 | OPimAccessTemplate<T>::~OPimAccessTemplate() { |
125 | qWarning("~OPimAccessTemplate<T>"); | 141 | qWarning("~OPimAccessTemplate<T>"); |
126 | delete m_backEnd; | 142 | delete m_backEnd; |
127 | } | 143 | } |
128 | template <class T> | 144 | template <class T> |
129 | bool OPimAccessTemplate<T>::load() { | 145 | bool OPimAccessTemplate<T>::load() { |
146 | invalidateCache(); | ||
130 | return m_backEnd->load(); | 147 | return m_backEnd->load(); |
131 | } | 148 | } |
132 | template <class T> | 149 | template <class T> |
133 | bool OPimAccessTemplate<T>::reload() { | 150 | bool OPimAccessTemplate<T>::reload() { |
134 | return m_backEnd->reload(); | 151 | return m_backEnd->reload(); |
135 | } | 152 | } |
136 | template <class T> | 153 | template <class T> |
137 | bool OPimAccessTemplate<T>::save() { | 154 | bool OPimAccessTemplate<T>::save() { |
138 | return m_backEnd->save(); | 155 | return m_backEnd->save(); |
139 | } | 156 | } |
140 | template <class T> | 157 | template <class T> |
141 | OPimAccessTemplate<T>::List OPimAccessTemplate<T>::allRecords()const { | 158 | OPimAccessTemplate<T>::List OPimAccessTemplate<T>::allRecords()const { |
142 | QArray<int> ints = m_backEnd->allRecords(); | 159 | QArray<int> ints = m_backEnd->allRecords(); |
143 | List lis(ints, this ); | 160 | List lis(ints, this ); |
144 | return lis; | 161 | return lis; |
145 | } | 162 | } |
146 | template <class T> | 163 | template <class T> |
147 | OPimAccessTemplate<T>::List | 164 | OPimAccessTemplate<T>::List |
148 | OPimAccessTemplate<T>::queryByExample( const T& t, int sortOrder ) { | 165 | OPimAccessTemplate<T>::queryByExample( const T& t, int sortOrder ) { |
149 | QArray<int> ints = m_backEnd->queryByExample( t, sortOrder ); | 166 | QArray<int> ints = m_backEnd->queryByExample( t, sortOrder ); |
150 | 167 | ||
151 | List lis(ints, this ); | 168 | List lis(ints, this ); |
152 | return lis; | 169 | return lis; |
153 | } | 170 | } |
154 | template <class T> | 171 | template <class T> |
155 | T OPimAccessTemplate<T>::find( int uid ) const{ | 172 | T OPimAccessTemplate<T>::find( int uid ) const{ |
156 | T t = m_backEnd->find( uid ); | 173 | T t = m_backEnd->find( uid ); |
174 | cache( t ); | ||
175 | return t; | ||
176 | } | ||
177 | template <class T> | ||
178 | T OPimAccessTemplate<T>::find( int uid, const QArray<int>& ar, | ||
179 | uint current, CacheDirection dir )const { | ||
180 | /* | ||
181 | * better do T.isEmpty() | ||
182 | * after a find this way we would | ||
183 | * avoid two finds in QCache... | ||
184 | */ | ||
185 | // qWarning("find it now %d", uid ); | ||
186 | if (m_cache.contains( uid ) ) { | ||
187 | qWarning("m cache contains %d", uid); | ||
188 | return m_cache.find( uid ); | ||
189 | } | ||
190 | |||
191 | T t = m_backEnd->find( uid, ar, current, dir ); | ||
192 | qWarning("found it and cache it now %d", uid); | ||
193 | cache( t ); | ||
157 | return t; | 194 | return t; |
158 | } | 195 | } |
159 | template <class T> | 196 | template <class T> |
160 | void OPimAccessTemplate<T>::clear() { | 197 | void OPimAccessTemplate<T>::clear() { |
161 | invalidateCache(); | 198 | invalidateCache(); |
162 | m_backEnd->clear(); | 199 | m_backEnd->clear(); |
163 | } | 200 | } |
164 | template <class T> | 201 | template <class T> |
165 | bool OPimAccessTemplate<T>::add( const T& t ) { | 202 | bool OPimAccessTemplate<T>::add( const T& t ) { |
203 | cache( t ); | ||
166 | return m_backEnd->add( t ); | 204 | return m_backEnd->add( t ); |
167 | } | 205 | } |
168 | template <class T> | 206 | template <class T> |
169 | bool OPimAccessTemplate<T>::remove( const T& t ) { | 207 | bool OPimAccessTemplate<T>::remove( const T& t ) { |
170 | return m_backEnd->remove( t.uid() ); | 208 | return remove( t.uid() ); |
171 | } | 209 | } |
172 | template <class T> | 210 | template <class T> |
173 | bool OPimAccessTemplate<T>::remove( int uid ) { | 211 | bool OPimAccessTemplate<T>::remove( int uid ) { |
212 | m_cache.remove( uid ); | ||
174 | return m_backEnd->remove( uid ); | 213 | return m_backEnd->remove( uid ); |
175 | } | 214 | } |
176 | template <class T> | 215 | template <class T> |
177 | bool OPimAccessTemplate<T>::replace( const T& t ) { | 216 | bool OPimAccessTemplate<T>::replace( const T& t ) { |
217 | m_cache.replace( t ); | ||
178 | return m_backEnd->replace( t ); | 218 | return m_backEnd->replace( t ); |
179 | } | 219 | } |
180 | template <class T> | 220 | template <class T> |
181 | void OPimAccessTemplate<T>::invalidateCache() { | 221 | void OPimAccessTemplate<T>::invalidateCache() { |
182 | 222 | m_cache.invalidate(); | |
183 | } | 223 | } |
184 | template <class T> | 224 | template <class T> |
185 | OPimAccessTemplate<T>::BackEnd* OPimAccessTemplate<T>::backEnd() { | 225 | OPimAccessTemplate<T>::BackEnd* OPimAccessTemplate<T>::backEnd() { |
186 | return m_backEnd; | 226 | return m_backEnd; |
187 | } | 227 | } |
188 | template <class T> | 228 | template <class T> |
189 | bool OPimAccessTemplate<T>::wasChangedExternally()const { | 229 | bool OPimAccessTemplate<T>::wasChangedExternally()const { |
190 | return false; | 230 | return false; |
191 | } | 231 | } |
192 | template <class T> | 232 | template <class T> |
193 | void OPimAccessTemplate<T>::setBackEnd( BackEnd* end ) { | 233 | void OPimAccessTemplate<T>::setBackEnd( BackEnd* end ) { |
194 | m_backEnd = end; | 234 | m_backEnd = end; |
235 | if (m_backEnd ) | ||
236 | m_backEnd->setFrontend( this ); | ||
237 | } | ||
238 | template <class T> | ||
239 | void OPimAccessTemplate<T>::cache( const T& t ) const{ | ||
240 | /* hacky we need to work around the const*/ | ||
241 | ((OPimAccessTemplate<T>*)this)->m_cache.add( t ); | ||
242 | } | ||
243 | template <class T> | ||
244 | void OPimAccessTemplate<T>::setSaneCacheSize( int size ) { | ||
245 | m_cache.setSize( size ); | ||
195 | } | 246 | } |
196 | #endif | 247 | #endif |
diff --git a/libopie2/opiepim/core/opimcache.h b/libopie2/opiepim/core/opimcache.h new file mode 100644 index 0000000..067f6e7 --- a/dev/null +++ b/libopie2/opiepim/core/opimcache.h | |||
@@ -0,0 +1,117 @@ | |||
1 | #ifndef OPIE_PIM_CACHE_H | ||
2 | #define OPIE_PIM_CACHE_H | ||
3 | |||
4 | #include <qintcache.h> | ||
5 | |||
6 | #include "opimrecord.h" | ||
7 | |||
8 | template <class T = OPimRecord> | ||
9 | class OPimCacheItem { | ||
10 | public: | ||
11 | OPimCacheItem( const T& t = T() ); | ||
12 | ~OPimCacheItem(); | ||
13 | |||
14 | T record()const; | ||
15 | void setRecord( const T& ); | ||
16 | private: | ||
17 | T m_t; | ||
18 | }; | ||
19 | |||
20 | /** | ||
21 | * OPimCache for caching the items | ||
22 | * We support adding, removing | ||
23 | * and finding | ||
24 | */ | ||
25 | template <class T = OPimRecord> | ||
26 | class OPimCache { | ||
27 | public: | ||
28 | typedef OPimCacheItem<T> Item; | ||
29 | OPimCache(); | ||
30 | ~OPimCache(); | ||
31 | |||
32 | bool contains(int uid)const; | ||
33 | void invalidate(); | ||
34 | void setSize( int size ); | ||
35 | |||
36 | T find(int uid )const; | ||
37 | void add( const T& ); | ||
38 | void remove( int uid ); | ||
39 | void replace( const T& ); | ||
40 | |||
41 | private: | ||
42 | QIntCache<Item> m_cache; | ||
43 | }; | ||
44 | |||
45 | // Implementation | ||
46 | template <class T> | ||
47 | OPimCacheItem<T>::OPimCacheItem( const T& t ) | ||
48 | : m_t(t) { | ||
49 | } | ||
50 | template <class T> | ||
51 | OPimCacheItem<T>::~OPimCacheItem() { | ||
52 | |||
53 | } | ||
54 | template <class T> | ||
55 | T OPimCacheItem<T>::record()const { | ||
56 | return m_t; | ||
57 | } | ||
58 | template <class T> | ||
59 | void OPimCacheItem<T>::setRecord( const T& t ) { | ||
60 | m_t = t; | ||
61 | } | ||
62 | // Cache | ||
63 | template <class T> | ||
64 | OPimCache<T>::OPimCache() { | ||
65 | m_cache.setAutoDelete( TRUE ); | ||
66 | } | ||
67 | template <class T> | ||
68 | OPimCache<T>::~OPimCache() { | ||
69 | |||
70 | } | ||
71 | template <class T> | ||
72 | bool OPimCache<T>::contains(int uid )const { | ||
73 | Item* it = m_cache.find( uid, FALSE ); | ||
74 | if (!it) | ||
75 | return false; | ||
76 | return true; | ||
77 | } | ||
78 | template <class T> | ||
79 | void OPimCache<T>::invalidate() { | ||
80 | m_cache.clear(); | ||
81 | } | ||
82 | template <class T> | ||
83 | void OPimCache<T>::setSize( int size ) { | ||
84 | m_cache.setMaxCost( size ); | ||
85 | } | ||
86 | template <class T> | ||
87 | T OPimCache<T>::find(int uid )const { | ||
88 | Item *it = m_cache.find( uid ); | ||
89 | if (it) | ||
90 | return it->record(); | ||
91 | return T(); | ||
92 | } | ||
93 | template <class T> | ||
94 | void OPimCache<T>::add( const T& t ) { | ||
95 | Item* it = 0l; | ||
96 | it = m_cache.find(t.uid(), FALSE ); | ||
97 | |||
98 | if (it ) | ||
99 | it->setRecord( t ); | ||
100 | |||
101 | it = new Item( t ); | ||
102 | if (!m_cache.insert( t.uid(), it ) ) | ||
103 | delete it; | ||
104 | } | ||
105 | template <class T> | ||
106 | void OPimCache<T>::remove( int uid ) { | ||
107 | m_cache.remove( uid ); | ||
108 | } | ||
109 | template <class T> | ||
110 | void OPimCache<T>::replace( const T& t) { | ||
111 | Item *it = m_cache.find( t.uid() ); | ||
112 | if ( it ) { | ||
113 | it->setRecord( t ); | ||
114 | } | ||
115 | } | ||
116 | |||
117 | #endif | ||
diff --git a/libopie2/opiepim/core/otemplatebase.h b/libopie2/opiepim/core/otemplatebase.h index f71417b..b855919 100644 --- a/libopie2/opiepim/core/otemplatebase.h +++ b/libopie2/opiepim/core/otemplatebase.h | |||
@@ -1,21 +1,32 @@ | |||
1 | #ifndef OPIE_TEMPLATE_BASE_H | 1 | #ifndef OPIE_TEMPLATE_BASE_H |
2 | #define OPIE_TEMPLATE_BASE_H | 2 | #define OPIE_TEMPLATE_BASE_H |
3 | 3 | ||
4 | #include <qarray.h> | ||
5 | |||
4 | #include "opimrecord.h" | 6 | #include "opimrecord.h" |
5 | 7 | ||
6 | /** | 8 | /** |
7 | * internal template base | 9 | * internal template base |
8 | */ | 10 | */ |
9 | template <class T = OPimRecord> | 11 | template <class T = OPimRecord> |
10 | class OTemplateBase { | 12 | class OTemplateBase { |
11 | public: | 13 | public: |
14 | enum CacheDirection { Forward=0, Reverse }; | ||
12 | OTemplateBase() { | 15 | OTemplateBase() { |
13 | }; | 16 | }; |
14 | virtual ~OTemplateBase() { | 17 | virtual ~OTemplateBase() { |
15 | } | 18 | } |
16 | virtual T find( int uid )const = 0; | 19 | virtual T find( int uid )const = 0; |
17 | 20 | ||
21 | /** | ||
22 | * read ahead find | ||
23 | */ | ||
24 | virtual T find( int uid, const QArray<int>& items, | ||
25 | uint current, CacheDirection dir = Forward )const = 0; | ||
26 | virtual void cache( const T& )const = 0; | ||
27 | virtual void setSaneCacheSize( int ) = 0; | ||
28 | |||
18 | }; | 29 | }; |
19 | 30 | ||
20 | 31 | ||
21 | #endif | 32 | #endif |
diff --git a/libopie2/opiepim/orecordlist.h b/libopie2/opiepim/orecordlist.h index b6fa7fa..08f5c85 100644 --- a/libopie2/opiepim/orecordlist.h +++ b/libopie2/opiepim/orecordlist.h | |||
@@ -55,32 +55,33 @@ public: | |||
55 | /** | 55 | /** |
56 | * the number of items | 56 | * the number of items |
57 | */ | 57 | */ |
58 | uint count()const; | 58 | uint count()const; |
59 | 59 | ||
60 | /** | 60 | /** |
61 | * sets the current item | 61 | * sets the current item |
62 | */ | 62 | */ |
63 | void setCurrent( uint cur ); | 63 | void setCurrent( uint cur ); |
64 | 64 | ||
65 | private: | 65 | private: |
66 | QArray<int> m_uids; | 66 | QArray<int> m_uids; |
67 | uint m_current; | 67 | uint m_current; |
68 | const Base* m_temp; | 68 | const Base* m_temp; |
69 | bool m_end : 1; | 69 | bool m_end : 1; |
70 | T m_record; | 70 | T m_record; |
71 | bool m_direction :1; | ||
71 | 72 | ||
72 | /* d pointer for future versions */ | 73 | /* d pointer for future versions */ |
73 | class IteratorPrivate; | 74 | class IteratorPrivate; |
74 | IteratorPrivate *d; | 75 | IteratorPrivate *d; |
75 | }; | 76 | }; |
76 | /** | 77 | /** |
77 | * The recordlist used as a return type | 78 | * The recordlist used as a return type |
78 | * from OPimAccessTemplate | 79 | * from OPimAccessTemplate |
79 | */ | 80 | */ |
80 | template <class T = OPimRecord > | 81 | template <class T = OPimRecord > |
81 | class ORecordList { | 82 | class ORecordList { |
82 | public: | 83 | public: |
83 | typedef OTemplateBase<T> Base; | 84 | typedef OTemplateBase<T> Base; |
84 | typedef ORecordListIterator<T> Iterator; | 85 | typedef ORecordListIterator<T> Iterator; |
85 | 86 | ||
86 | /** | 87 | /** |
@@ -112,115 +113,119 @@ public: | |||
112 | /* | 113 | /* |
113 | ConstIterator begin()const; | 114 | ConstIterator begin()const; |
114 | ConstIterator end()const; | 115 | ConstIterator end()const; |
115 | */ | 116 | */ |
116 | private: | 117 | private: |
117 | QArray<int> m_ids; | 118 | QArray<int> m_ids; |
118 | const Base* m_acc; | 119 | const Base* m_acc; |
119 | }; | 120 | }; |
120 | 121 | ||
121 | /* ok now implement it */ | 122 | /* ok now implement it */ |
122 | template <class T> | 123 | template <class T> |
123 | ORecordListIterator<T>::ORecordListIterator() { | 124 | ORecordListIterator<T>::ORecordListIterator() { |
124 | m_current = 0; | 125 | m_current = 0; |
125 | m_temp = 0l; | 126 | m_temp = 0l; |
126 | m_end = true; | 127 | m_end = true; |
127 | m_record = T(); | 128 | m_record = T(); |
129 | /* forward */ | ||
130 | m_direction = TRUE; | ||
128 | } | 131 | } |
129 | template <class T> | 132 | template <class T> |
130 | ORecordListIterator<T>::~ORecordListIterator() { | 133 | ORecordListIterator<T>::~ORecordListIterator() { |
131 | /* nothing to delete */ | 134 | /* nothing to delete */ |
132 | } | 135 | } |
133 | 136 | ||
134 | template <class T> | 137 | template <class T> |
135 | ORecordListIterator<T>::ORecordListIterator( const ORecordListIterator<T>& it) { | 138 | ORecordListIterator<T>::ORecordListIterator( const ORecordListIterator<T>& it) { |
136 | // qWarning("ORecordListIterator copy c'tor"); | 139 | // qWarning("ORecordListIterator copy c'tor"); |
137 | m_uids = it.m_uids; | 140 | m_uids = it.m_uids; |
138 | m_current = it.m_current; | 141 | m_current = it.m_current; |
139 | m_temp = it.m_temp; | 142 | m_temp = it.m_temp; |
140 | m_end = it.m_end; | 143 | m_end = it.m_end; |
141 | m_record = it.m_record; | 144 | m_record = it.m_record; |
145 | m_direction = it.m_direction; | ||
142 | } | 146 | } |
143 | 147 | ||
144 | template <class T> | 148 | template <class T> |
145 | ORecordListIterator<T> &ORecordListIterator<T>::operator=( const ORecordListIterator<T>& it) { | 149 | ORecordListIterator<T> &ORecordListIterator<T>::operator=( const ORecordListIterator<T>& it) { |
146 | m_uids = it.m_uids; | 150 | m_uids = it.m_uids; |
147 | m_current = it.m_current; | 151 | m_current = it.m_current; |
148 | m_temp = it.m_temp; | 152 | m_temp = it.m_temp; |
149 | m_end = it.m_end; | 153 | m_end = it.m_end; |
150 | m_record = it.m_record; | 154 | m_record = it.m_record; |
151 | 155 | ||
152 | return *this; | 156 | return *this; |
153 | } | 157 | } |
154 | 158 | ||
155 | template <class T> | 159 | template <class T> |
156 | T ORecordListIterator<T>::operator*() { | 160 | T ORecordListIterator<T>::operator*() { |
157 | qWarning("operator* %d %d", m_current, m_uids[m_current] ); | 161 | qWarning("operator* %d %d", m_current, m_uids[m_current] ); |
158 | if (!m_end ) | 162 | if (!m_end ) |
159 | /* FIXME | 163 | m_record = m_temp->find( m_uids[m_current], m_uids, m_current, |
160 | * until the cache is in place | 164 | m_direction ? Base::Forward : |
161 | * we do the uid match uid check | 165 | Base::Reverse ); |
162 | */ | ||
163 | m_record = m_temp->find( m_uids[m_current] ); | ||
164 | else | 166 | else |
165 | m_record = T(); | 167 | m_record = T(); |
166 | 168 | ||
167 | return m_record; | 169 | return m_record; |
168 | } | 170 | } |
169 | 171 | ||
170 | template <class T> | 172 | template <class T> |
171 | ORecordListIterator<T> &ORecordListIterator<T>::operator++() { | 173 | ORecordListIterator<T> &ORecordListIterator<T>::operator++() { |
174 | m_direction = true; | ||
172 | if (m_current < m_uids.count() ) { | 175 | if (m_current < m_uids.count() ) { |
173 | m_end = false; | 176 | m_end = false; |
174 | ++m_current; | 177 | ++m_current; |
175 | }else | 178 | }else |
176 | m_end = true; | 179 | m_end = true; |
177 | 180 | ||
178 | return *this; | 181 | return *this; |
179 | } | 182 | } |
180 | template <class T> | 183 | template <class T> |
181 | ORecordListIterator<T> &ORecordListIterator<T>::operator--() { | 184 | ORecordListIterator<T> &ORecordListIterator<T>::operator--() { |
185 | m_direction = false; | ||
182 | if ( m_current > 0 ) { | 186 | if ( m_current > 0 ) { |
183 | --m_current; | 187 | --m_current; |
184 | m_end = false; | 188 | m_end = false; |
185 | } else | 189 | } else |
186 | m_end = true; | 190 | m_end = true; |
187 | 191 | ||
188 | return *this; | 192 | return *this; |
189 | } | 193 | } |
190 | 194 | ||
191 | template <class T> | 195 | template <class T> |
192 | bool ORecordListIterator<T>::operator==( const ORecordListIterator<T>& it ) { | 196 | bool ORecordListIterator<T>::operator==( const ORecordListIterator<T>& it ) { |
193 | 197 | ||
194 | /* if both are at we're the same.... */ | 198 | /* if both are at we're the same.... */ |
195 | if ( m_end == it.m_end ) return true; | 199 | if ( m_end == it.m_end ) return true; |
196 | 200 | ||
197 | if ( m_uids != it.m_uids ) return false; | 201 | if ( m_uids != it.m_uids ) return false; |
198 | if ( m_current != it.m_current ) return false; | 202 | if ( m_current != it.m_current ) return false; |
199 | if ( m_temp != it.m_temp ) return false; | 203 | if ( m_temp != it.m_temp ) return false; |
200 | 204 | ||
201 | return true; | 205 | return true; |
202 | } | 206 | } |
203 | template <class T> | 207 | template <class T> |
204 | bool ORecordListIterator<T>::operator!=( const ORecordListIterator<T>& it ) { | 208 | bool ORecordListIterator<T>::operator!=( const ORecordListIterator<T>& it ) { |
205 | return !(*this == it ); | 209 | return !(*this == it ); |
206 | } | 210 | } |
207 | template <class T> | 211 | template <class T> |
208 | ORecordListIterator<T>::ORecordListIterator( const QArray<int> uids, | 212 | ORecordListIterator<T>::ORecordListIterator( const QArray<int> uids, |
209 | const Base* t ) | 213 | const Base* t ) |
210 | : m_uids( uids ), m_current( 0 ), m_temp( t ), m_end( false ) | 214 | : m_uids( uids ), m_current( 0 ), m_temp( t ), m_end( false ), |
215 | m_direction( false ) | ||
211 | { | 216 | { |
212 | } | 217 | } |
213 | template <class T> | 218 | template <class T> |
214 | uint ORecordListIterator<T>::current()const { | 219 | uint ORecordListIterator<T>::current()const { |
215 | return m_current; | 220 | return m_current; |
216 | } | 221 | } |
217 | template <class T> | 222 | template <class T> |
218 | void ORecordListIterator<T>::setCurrent( uint cur ) { | 223 | void ORecordListIterator<T>::setCurrent( uint cur ) { |
219 | if( cur < m_uids.count() ) { | 224 | if( cur < m_uids.count() ) { |
220 | m_end = false; | 225 | m_end = false; |
221 | m_current= cur; | 226 | m_current= cur; |
222 | } | 227 | } |
223 | } | 228 | } |
224 | template <class T> | 229 | template <class T> |
225 | uint ORecordListIterator<T>::count()const { | 230 | uint ORecordListIterator<T>::count()const { |
226 | return m_uids.count(); | 231 | return m_uids.count(); |
@@ -241,19 +246,20 @@ ORecordList<T>::Iterator ORecordList<T>::begin() { | |||
241 | return it; | 246 | return it; |
242 | } | 247 | } |
243 | template <class T> | 248 | template <class T> |
244 | ORecordList<T>::Iterator ORecordList<T>::end() { | 249 | ORecordList<T>::Iterator ORecordList<T>::end() { |
245 | Iterator it( m_ids, m_acc ); | 250 | Iterator it( m_ids, m_acc ); |
246 | it.m_end = true; | 251 | it.m_end = true; |
247 | it.m_current = m_ids.count(); | 252 | it.m_current = m_ids.count(); |
248 | 253 | ||
249 | return it; | 254 | return it; |
250 | } | 255 | } |
251 | template <class T> | 256 | template <class T> |
252 | uint ORecordList<T>::count()const { | 257 | uint ORecordList<T>::count()const { |
253 | return m_ids.count(); | 258 | return m_ids.count(); |
254 | } | 259 | } |
255 | template <class T> | 260 | template <class T> |
256 | T ORecordList<T>::operator[]( uint i ) { | 261 | T ORecordList<T>::operator[]( uint i ) { |
257 | return m_acc->find( m_ids[i] ); | 262 | /* forward */ |
263 | return m_acc->find( m_ids[i], m_ids, i ); | ||
258 | } | 264 | } |
259 | #endif | 265 | #endif |