1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_extensions.hxx"
26 #include "fieldmappingimpl.hxx"
27 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
28 #include <com/sun/star/beans/PropertyValue.hpp>
29 #include <com/sun/star/beans/XPropertySet.hpp>
30 #include <com/sun/star/ui/dialogs/XExecutableDialog.hpp>
31 #include <com/sun/star/awt/XWindow.hpp>
32 #include <com/sun/star/sdb/CommandType.hpp>
33 #include <tools/debug.hxx>
34 #ifndef _TOOLKIT_HELPER_VCLUNOHELPER_HXX_
35 #include <toolkit/unohlp.hxx>
36 #endif
37 #include <vcl/stdtext.hxx>
38 #include <com/sun/star/util/AliasProgrammaticPair.hpp>
39 #ifndef EXTENSIONS_ABPRESID_HRC
40 #include "abpresid.hrc"
41 #endif
42 #include "componentmodule.hxx"
43 #include <unotools/confignode.hxx>
44 
45 //.........................................................................
46 namespace abp
47 {
48 //.........................................................................
49 
50 	using namespace ::utl;
51 	using namespace ::com::sun::star::uno;
52 	using namespace ::com::sun::star::awt;
53 	using namespace ::com::sun::star::util;
54 	using namespace ::com::sun::star::lang;
55 	using namespace ::com::sun::star::beans;
56 	using namespace ::com::sun::star::sdb;
57 	using namespace ::com::sun::star::ui::dialogs;
58 
59 	//---------------------------------------------------------------------
60 	static const ::rtl::OUString& lcl_getDriverSettingsNodeName()
61 	{
62 		static const ::rtl::OUString s_sDriverSettingsNodeName =
63 			::rtl::OUString::createFromAscii( "/org.openoffice.Office.DataAccess/DriverSettings/com.sun.star.comp.sdbc.MozabDriver" );
64 		return s_sDriverSettingsNodeName;
65 	}
66 
67 	//---------------------------------------------------------------------
68 	static const ::rtl::OUString& lcl_getAddressBookNodeName()
69 	{
70 		static const ::rtl::OUString s_sAddressBookNodeName =
71 			::rtl::OUString::createFromAscii( "/org.openoffice.Office.DataAccess/AddressBook" );
72 		return s_sAddressBookNodeName;
73 	}
74 
75 	//.....................................................................
76 	namespace fieldmapping
77 	{
78 	//.....................................................................
79 
80 		//-----------------------------------------------------------------
81 		sal_Bool invokeDialog( const Reference< XMultiServiceFactory >& _rxORB, class Window* _pParent,
82 			const Reference< XPropertySet >& _rxDataSource, AddressSettings& _rSettings ) SAL_THROW ( ( ) )
83 		{
84             _rSettings.aFieldMapping.clear();
85 
86 			DBG_ASSERT( _rxORB.is(), "fieldmapping::invokeDialog: invalid service factory!" );
87 			DBG_ASSERT( _rxDataSource.is(), "fieldmapping::invokeDialog: invalid data source!" );
88             if ( !_rxORB.is() || !_rxDataSource.is() )
89                 return sal_False;
90 
91 			try
92 			{
93 				// ........................................................
94 				// the parameters for creating the dialog
95 				Sequence< Any > aArguments(5);
96 				Any* pArguments = aArguments.getArray();
97 
98 				// the parent window
99 				Reference< XWindow > xDialogParent = VCLUnoHelper::GetInterface( _pParent );
100 				*pArguments++ <<= PropertyValue(::rtl::OUString::createFromAscii( "ParentWindow" ), -1, makeAny( xDialogParent ), PropertyState_DIRECT_VALUE);
101 
102 				// the data source to use
103 				*pArguments++ <<= PropertyValue(::rtl::OUString::createFromAscii( "DataSource" ), -1, makeAny( _rxDataSource ), PropertyState_DIRECT_VALUE);
104                 *pArguments++ <<= PropertyValue(::rtl::OUString::createFromAscii( "DataSourceName" ), -1, makeAny( (sal_Bool)_rSettings.bRegisterDataSource ? _rSettings.sRegisteredDataSourceName : _rSettings.sDataSourceName ), PropertyState_DIRECT_VALUE);
105 
106 				// the table to use
107 				*pArguments++ <<= PropertyValue(::rtl::OUString::createFromAscii( "Command" ), -1, makeAny( _rSettings.sSelectedTable ), PropertyState_DIRECT_VALUE);
108 
109 				// the title
110 				::rtl::OUString sTitle = String( ModuleRes( RID_STR_FIELDDIALOGTITLE ) );
111 				*pArguments++ <<= PropertyValue(::rtl::OUString::createFromAscii( "Title" ), -1, makeAny( sTitle ), PropertyState_DIRECT_VALUE);
112 
113 				// ........................................................
114 				// create an instance of the dialog service
115 				static ::rtl::OUString s_sAdressBookFieldAssignmentServiceName = ::rtl::OUString::createFromAscii( "com.sun.star.ui.AddressBookSourceDialog" );
116 				Reference< XExecutableDialog > xDialog(
117 					_rxORB->createInstanceWithArguments( s_sAdressBookFieldAssignmentServiceName, aArguments ),
118 					UNO_QUERY
119 				);
120 				if ( !xDialog.is( ) )
121 				{
122 					ShowServiceNotAvailableError( _pParent, s_sAdressBookFieldAssignmentServiceName, sal_True );
123 					return sal_False;
124 				}
125 
126 				// execute the dialog
127 				if ( xDialog->execute() )
128 				{
129 					// retrieve the field mapping as set by he user
130 					Reference< XPropertySet > xDialogProps( xDialog, UNO_QUERY );
131 
132 					Sequence< AliasProgrammaticPair > aMapping;
133 #ifdef DBG_UTIL
134 					sal_Bool bSuccess =
135 #endif
136 					xDialogProps->getPropertyValue( ::rtl::OUString::createFromAscii( "FieldMapping" ) ) >>= aMapping;
137 					DBG_ASSERT( bSuccess, "fieldmapping::invokeDialog: invalid property type for FieldMapping!" );
138 
139 					// and copy it into the map
140 					const AliasProgrammaticPair* pMapping = aMapping.getConstArray();
141 					const AliasProgrammaticPair* pMappingEnd = pMapping + aMapping.getLength();
142 					for (;pMapping != pMappingEnd; ++pMapping)
143 						_rSettings.aFieldMapping[ pMapping->ProgrammaticName ] = pMapping->Alias;
144 
145 					return sal_True;
146 				}
147 
148 			}
149 			catch(const Exception&)
150 			{
151 				DBG_ERROR("fieldmapping::invokeDialog: caught an exception while executing the dialog!");
152 			}
153 			return sal_False;
154 		}
155 
156 		//-----------------------------------------------------------------
157 		void defaultMapping(  const Reference< XMultiServiceFactory >& _rxORB, MapString2String& _rFieldAssignment ) SAL_THROW ( ( ) )
158 		{
159 			_rFieldAssignment.clear();
160 
161 			try
162 			{
163 				// what we have:
164 				// a) For the address data source, we need a mapping from programmatic names (1) to real column names
165 				// b) The SDBC driver has a fixed set of columns, which, when returned, are named according to
166 				//    some configuration entries. E.g., the driver displays the field which it knows contains
167 				//    the first name as "First Name" - the latter string is stored in the config.
168 				//    For this, the driver uses programmatic names, too, but they differ from the programmatic names the
169 				//    template documents have.
170 				// So what we need first is a mapping from programmatic names (1) to programmatic names (2)
171 				const sal_Char* pMappingProgrammatics[] =
172 				{
173 					"FirstName",			"FirstName",
174 					"LastName",				"LastName",
175 					"Street",				"HomeAddress",
176 					"Zip",					"HomeZipCode",
177 					"City",					"HomeCity",
178 					"State",				"HomeState",
179 					"Country",				"HomeCountry",
180 					"PhonePriv",			"HomePhone",
181 					"PhoneComp",			"WorkPhone",
182 					"PhoneCell",			"CellularNumber",
183 					"Pager",				"PagerNumber",
184 					"Fax",					"FaxNumber",
185 					"EMail",				"PrimaryEmail",
186 					"URL",					"WebPage1",
187 					"Note",					"Notes",
188 					"Altfield1",			"Custom1",
189 					"Altfield2",			"Custom2",
190 					"Altfield3",			"Custom3",
191 					"Altfield4",			"Custom4",
192 					"Title",				"JobTitle",
193 					"Company",				"Company",
194 					"Department",			"Department"
195 				};
196 					// (this list is not complete: both lists of programmatic names are larger in real,
197 					// but this list above is the intersection)
198 
199 
200 				// access the configuration information which the driver uses for determining it's column names
201 				::rtl::OUString sDriverAliasesNodeName = lcl_getDriverSettingsNodeName();
202 				sDriverAliasesNodeName += ::rtl::OUString::createFromAscii( "/ColumnAliases" );
203 
204 				// create a config node for this
205 				OConfigurationTreeRoot aDriverFieldAliasing = OConfigurationTreeRoot::createWithServiceFactory(
206 					_rxORB, sDriverAliasesNodeName, -1, OConfigurationTreeRoot::CM_READONLY);
207 
208 				// loop through all programmatic pairs
209 				DBG_ASSERT( 0 == ( sizeof( pMappingProgrammatics ) / sizeof( pMappingProgrammatics[ 0 ] ) ) % 2,
210 					"fieldmapping::defaultMapping: invalid programmatic map!" );
211 				// number of pairs
212 				sal_Int32 nIntersectedProgrammatics = sizeof( pMappingProgrammatics ) / sizeof( pMappingProgrammatics[ 0 ] ) / 2;
213 
214 				const sal_Char** pProgrammatic = pMappingProgrammatics;
215 				::rtl::OUString sAddressProgrammatic;
216 				::rtl::OUString sDriverProgrammatic;
217 				::rtl::OUString sDriverUI;
218 				for	(	sal_Int32 i=0;
219 						i < nIntersectedProgrammatics;
220 						++i
221 					)
222 				{
223 					sAddressProgrammatic = ::rtl::OUString::createFromAscii( *pProgrammatic++ );
224 					sDriverProgrammatic = ::rtl::OUString::createFromAscii( *pProgrammatic++ );
225 
226 					if ( aDriverFieldAliasing.hasByName( sDriverProgrammatic ) )
227 					{
228 						aDriverFieldAliasing.getNodeValue( sDriverProgrammatic ) >>= sDriverUI;
229 						if ( 0 == sDriverUI.getLength() )
230 						{
231 							DBG_ERROR( "fieldmapping::defaultMapping: invalid driver UI column name!");
232 						}
233 						else
234 							_rFieldAssignment[ sAddressProgrammatic ] = sDriverUI;
235 					}
236 					else
237 					{
238 						DBG_ERROR( "fieldmapping::defaultMapping: invalid driver programmatic name!" );
239 					}
240 				}
241 			}
242 			catch( const Exception& )
243 			{
244 				DBG_ERROR("fieldmapping::defaultMapping: code is assumed to throw no exceptions!");
245 					// the config nodes we're using herein should not do this ....
246 			}
247 		}
248 
249 		//-----------------------------------------------------------------
250 		void writeTemplateAddressFieldMapping( const Reference< XMultiServiceFactory >& _rxORB, const MapString2String& _rFieldAssignment ) SAL_THROW ( ( ) )
251 		{
252 			// want to have a non-const map for easier handling
253 			MapString2String aFieldAssignment( _rFieldAssignment );
254 
255 			// access the configuration information which the driver uses for determining it's column names
256 			const ::rtl::OUString& sAddressBookNodeName = lcl_getAddressBookNodeName();
257 
258 			// create a config node for this
259 			OConfigurationTreeRoot aAddressBookSettings = OConfigurationTreeRoot::createWithServiceFactory(
260 				_rxORB, sAddressBookNodeName, -1, OConfigurationTreeRoot::CM_UPDATABLE);
261 
262 			OConfigurationNode aFields = aAddressBookSettings.openNode( ::rtl::OUString::createFromAscii( "Fields" ) );
263 
264 			// loop through all existent fields
265 			Sequence< ::rtl::OUString > aExistentFields = aFields.getNodeNames();
266 			const ::rtl::OUString* pExistentFields = aExistentFields.getConstArray();
267 			const ::rtl::OUString* pExistentFieldsEnd = pExistentFields + aExistentFields.getLength();
268 
269 			const ::rtl::OUString sProgrammaticNodeName = ::rtl::OUString::createFromAscii( "ProgrammaticFieldName" );
270 			const ::rtl::OUString sAssignedNodeName = ::rtl::OUString::createFromAscii( "AssignedFieldName" );
271 
272 			for ( ; pExistentFields != pExistentFieldsEnd; ++pExistentFields )
273 			{
274 #ifdef DBG_UTIL
275 				::rtl::OUString sRedundantProgrammaticName;
276 				aFields.openNode( *pExistentFields ).getNodeValue( sProgrammaticNodeName ) >>= sRedundantProgrammaticName;
277 #endif
278 				DBG_ASSERT( sRedundantProgrammaticName == *pExistentFields,
279 					"fieldmapping::writeTemplateAddressFieldMapping: inconsistent config data!" );
280 					// there should be a redundancy in the config data .... if this asserts, there isn't anymore!
281 
282 				// do we have a new alias for the programmatic?
283 				MapString2StringIterator aPos = aFieldAssignment.find( *pExistentFields );
284 				if ( aFieldAssignment.end() != aPos )
285 				{	// yes
286 					// -> set a new value
287 					OConfigurationNode aExistentField = aFields.openNode( *pExistentFields );
288 					aExistentField.setNodeValue( sAssignedNodeName, makeAny( aPos->second ) );
289 					// and remove the mapping entry
290 					aFieldAssignment.erase( *pExistentFields );
291 				}
292 				else
293 				{	// no
294 					// -> remove it
295 					aFields.removeNode( *pExistentFields );
296 				}
297 			}
298 
299 			// now everything remaining in aFieldAssignment marks a mapping entry which was not present
300 			// in the config before
301 			for (	ConstMapString2StringIterator aNewMapping = aFieldAssignment.begin();
302 					aNewMapping != aFieldAssignment.end();
303 					++aNewMapping
304 				)
305 			{
306 				DBG_ASSERT( !aFields.hasByName( aNewMapping->first ),
307 					"fieldmapping::writeTemplateAddressFieldMapping: inconsistence!" );
308 					// in case the config node for the fields already has the node named <aNewMapping->first>,
309 					// the entry should have been removed from aNewMapping (in the above loop)
310 				OConfigurationNode aNewField =  aFields.createNode( aNewMapping->first );
311 				aNewField.setNodeValue( sProgrammaticNodeName, makeAny( aNewMapping->first ) );
312 				aNewField.setNodeValue( sAssignedNodeName, makeAny( aNewMapping->second ) );
313 			}
314 
315 			// commit the changes done
316 			aAddressBookSettings.commit();
317 		}
318 
319 	//.....................................................................
320 	}	// namespace fieldmapping
321 	//.....................................................................
322 
323 	//.....................................................................
324 	namespace addressconfig
325 	{
326 	//.....................................................................
327 
328 		//-----------------------------------------------------------------
329 		void writeTemplateAddressSource( const Reference< XMultiServiceFactory >& _rxORB,
330 			const ::rtl::OUString& _rDataSourceName, const ::rtl::OUString& _rTableName ) SAL_THROW ( ( ) )
331 		{
332 			// access the configuration information which the driver uses for determining it's column names
333 			const ::rtl::OUString& sAddressBookNodeName = lcl_getAddressBookNodeName();
334 
335 			// create a config node for this
336 			OConfigurationTreeRoot aAddressBookSettings = OConfigurationTreeRoot::createWithServiceFactory(
337 				_rxORB, sAddressBookNodeName, -1, OConfigurationTreeRoot::CM_UPDATABLE);
338 
339 			aAddressBookSettings.setNodeValue( ::rtl::OUString::createFromAscii( "DataSourceName" ), makeAny( _rDataSourceName ) );
340 			aAddressBookSettings.setNodeValue( ::rtl::OUString::createFromAscii( "Command" ), makeAny( _rTableName ) );
341 			aAddressBookSettings.setNodeValue( ::rtl::OUString::createFromAscii( "CommandType" ), makeAny( (sal_Int32)CommandType::TABLE ) );
342 
343 			// commit the changes done
344 			aAddressBookSettings.commit();
345 		}
346 
347 		//-----------------------------------------------------------------
348 		void markPilotSuccess( const Reference< XMultiServiceFactory >& _rxORB ) SAL_THROW ( ( ) )
349 		{
350 			// access the configuration information which the driver uses for determining it's column names
351 			const ::rtl::OUString& sAddressBookNodeName = lcl_getAddressBookNodeName();
352 
353 			// create a config node for this
354 			OConfigurationTreeRoot aAddressBookSettings = OConfigurationTreeRoot::createWithServiceFactory(
355 				_rxORB, sAddressBookNodeName, -1, OConfigurationTreeRoot::CM_UPDATABLE);
356 
357 			// set the flag
358 			aAddressBookSettings.setNodeValue( ::rtl::OUString::createFromAscii( "AutoPilotCompleted" ), makeAny( (sal_Bool)sal_True ) );
359 
360 			// commit the changes done
361 			aAddressBookSettings.commit();
362 		}
363 
364 	//.....................................................................
365 	}	// namespace addressconfig
366 	//.....................................................................
367 
368 //.........................................................................
369 }	// namespace abp
370 //.........................................................................
371 
372