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 //_______________________________________________ 25 // imports 26 import com.sun.star.uno.XComponentContext; 27 import com.sun.star.lib.uno.helper.Factory; 28 import com.sun.star.lib.uno.helper.WeakBase; 29 import com.sun.star.lang.XServiceInfo; 30 import com.sun.star.awt.*; 31 import com.sun.star.beans.*; 32 import com.sun.star.task.*; 33 import com.sun.star.uno.*; 34 import java.lang.*; 35 import javax.swing.*; 36 37 //_______________________________________________ 38 // implementation 39 40 /** it implements a simple job component. 41 * 42 * Such jobs are executable in different ways: 43 * <ul> 44 * <li>registered for a special URL schema "vnd.sun.star.jobs:*" and used from the generic dispatch framework</li> 45 * <li>the global com.sun.star.task.JobExecutor service and registered for special events.</li> 46 * </ul> 47 */ 48 public class AsyncJob extends WeakBase implements XServiceInfo, XAsyncJob 49 { 50 //___________________________________________ 51 // const 52 public final XComponentContext m_xCmpCtx; 53 54 /** the const list of supported uno service names. */ 55 public static final java.lang.String[] SERVICENAMES = {"com.sun.star.task.AsyncJob"}; 56 57 /** the const uno implementation name. 58 * It must be an unique value! The best naming schema seems to use 59 * a registered domain in reverse order ... 60 */ 61 public static final java.lang.String IMPLEMENTATIONNAME = "com.sun.star.comp.framework.java.services.AsyncJob"; 62 63 //___________________________________________ 64 // interface 65 66 /** initialize a new instance of this class with default values. */ 67 public AsyncJob( XComponentContext xCompContext ) 68 { 69 m_xCmpCtx = xCompContext; 70 } 71 72 //___________________________________________ 73 74 /** starts execution of this job. 75 * 76 * @param lArgs 77 * list which contains: 78 * <ul> 79 * <li>generic job configuration data</li> 80 * <li>job specific configuration data</li> 81 * <li>some environment informations</li> 82 * <li>may optional arguments of a corresponding dispatch request</li> 83 * </ul> 84 * 85 * @params xListener 86 * callback to the executor of this job, which control our life time 87 * 88 * @throws com.sun.star.lang.IllegalArgumentException 89 * if given argument list seems to be wrong 90 */ 91 public synchronized void executeAsync(com.sun.star.beans.NamedValue[] lArgs , 92 com.sun.star.task.XJobListener xListener) 93 throws com.sun.star.lang.IllegalArgumentException 94 { 95 // For asynchronous jobs a valid listener reference is guaranteed normally ... 96 if (xListener == null) 97 throw new com.sun.star.lang.IllegalArgumentException("invalid listener"); 98 99 // extract all possible sub list of given argument list 100 com.sun.star.beans.NamedValue[] lGenericConfig = null; 101 com.sun.star.beans.NamedValue[] lJobConfig = null; 102 com.sun.star.beans.NamedValue[] lEnvironment = null; 103 com.sun.star.beans.NamedValue[] lDynamicData = null; 104 105 int c = lArgs.length; 106 for (int i=0; i<c; ++i) 107 { 108 if (lArgs[i].Name.equals("Config")) 109 lGenericConfig = (com.sun.star.beans.NamedValue[])com.sun.star.uno.AnyConverter.toArray(lArgs[i].Value); 110 else 111 if (lArgs[i].Name.equals("JobConfig")) 112 lJobConfig = (com.sun.star.beans.NamedValue[])com.sun.star.uno.AnyConverter.toArray(lArgs[i].Value); 113 else 114 if (lArgs[i].Name.equals("Environment")) 115 lEnvironment = (com.sun.star.beans.NamedValue[])com.sun.star.uno.AnyConverter.toArray(lArgs[i].Value); 116 else 117 if (lArgs[i].Name.equals("DynamicData")) 118 lDynamicData = (com.sun.star.beans.NamedValue[])com.sun.star.uno.AnyConverter.toArray(lArgs[i].Value); 119 } 120 121 // Analyze the environment info. This sub list is the only guaranteed one! 122 if (lEnvironment == null) 123 throw new com.sun.star.lang.IllegalArgumentException("no environment"); 124 125 java.lang.String sEnvType = null; 126 java.lang.String sEventName = null; 127 com.sun.star.frame.XFrame xFrame = null; 128 c = lEnvironment.length; 129 for (int i=0; i<c; ++i) 130 { 131 if (lEnvironment[i].Name.equals("EnvType")) 132 sEnvType = com.sun.star.uno.AnyConverter.toString(lEnvironment[i].Value); 133 else 134 if (lEnvironment[i].Name.equals("EventName")) 135 sEventName = com.sun.star.uno.AnyConverter.toString(lEnvironment[i].Value); 136 else 137 if (lEnvironment[i].Name.equals("Frame")) 138 xFrame = (com.sun.star.frame.XFrame)com.sun.star.uno.AnyConverter.toObject( 139 new com.sun.star.uno.Type(com.sun.star.frame.XFrame.class), 140 lEnvironment[i].Value); 141 } 142 143 // Further the environment property "EnvType" is required as minimum. 144 if ( 145 (sEnvType==null) || 146 ( 147 (!sEnvType.equals("EXECUTOR")) && 148 (!sEnvType.equals("DISPATCH")) 149 ) 150 ) 151 { 152 java.lang.String sMessage = "\"" + sEnvType + "\" isn't a valid value for EnvType"; 153 throw new com.sun.star.lang.IllegalArgumentException(sMessage); 154 } 155 156 // Analyze the set of shared config data. 157 java.lang.String sAlias = null; 158 if (lGenericConfig!=null) 159 { 160 c = lGenericConfig.length; 161 for (int i=0; i<c; ++i) 162 { 163 if (lGenericConfig[i].Name.equals("Alias")) 164 sAlias = com.sun.star.uno.AnyConverter.toString(lGenericConfig[i].Value); 165 } 166 } 167 168 // do your job ... 169 // Here we print out all found arguments. 170 java.lang.String sOut = formatOutArgs(lGenericConfig, lJobConfig, lEnvironment, lDynamicData); 171 if (xFrame != null) 172 showInfoModal(xFrame.getContainerWindow(), "Arguments of AsyncJob initialization ...", sOut); 173 else 174 showInfoNonModal("Arguments of AsyncJob initialization ...", sOut); 175 176 // use return value to start different actions 177 // But look for the right environment. Some options make no sense inside the wrong env. 178 com.sun.star.beans.NamedValue aDeactivation = null; 179 com.sun.star.beans.NamedValue aDispatchResult = null; 180 com.sun.star.beans.NamedValue aSaveRequest = null; 181 182 // SaveArguments will be made everytimes! 183 c = 1; 184 185 if (lJobConfig==null) 186 lJobConfig = new com.sun.star.beans.NamedValue[1]; 187 lJobConfig[0] = new com.sun.star.beans.NamedValue(); 188 lJobConfig[0].Name = "arg_1"; 189 lJobConfig[0].Value = "val_1"; 190 191 aSaveRequest = new com.sun.star.beans.NamedValue(); 192 aSaveRequest.Name = "SaveArguments"; 193 aSaveRequest.Value = lJobConfig; 194 195 // Deactivation is useful inside EXECUTOR environment only 196 if (sEnvType.equals("EXECUTOR")) 197 { 198 ++c; 199 aDeactivation = new com.sun.star.beans.NamedValue(); 200 aDeactivation.Name = "Deactivate"; 201 aDeactivation.Value = java.lang.Boolean.TRUE; 202 } 203 204 // Sending of result events is useful inside DISPATCH environment only 205 if (sEnvType.equals("DISPATCH")) 206 { 207 ++c; 208 aDispatchResult = new com.sun.star.beans.NamedValue(); 209 aDispatchResult.Name = "SendDispatchResult"; 210 aDispatchResult.Value = new com.sun.star.frame.DispatchResultEvent(this, com.sun.star.frame.DispatchResultState.SUCCESS, null); 211 } 212 213 // pack it together for return 214 int i=0; 215 com.sun.star.beans.NamedValue[] lReturn = new com.sun.star.beans.NamedValue[c]; 216 lReturn[i++] = aSaveRequest; 217 if (aDeactivation!=null) 218 lReturn[i++] = aDeactivation; 219 if (aDispatchResult!=null) 220 lReturn[i++] = aDispatchResult; 221 222 xListener.jobFinished(this, lReturn); 223 } 224 225 //___________________________________________ 226 227 /** show an info box with the UNO based toolkit. 228 * 229 * It tries to use the container window of a may well know 230 * office frame as parent. If such parent window could be located, 231 * the info box can be shown in modal mode. If a parent is missing 232 * (because this job is called inside an EXECUTOR environment, which 233 * does not set any frame context here) the info box can't be created! 234 * Because the toolkit needs parents for non top level windows ... 235 * In that case the only way is to implement this info box 236 * native or make it non modal using java dialogs inside it's own thread ... 237 * (see showInfoNonModal() too) 238 * 239 * @param xParent 240 * used as parent window of the shown info box. 241 * 242 * @param sTitle 243 * is shown as title of the info box. 244 * 245 * @param sMessage 246 * inclused the message body, which is shown as info. 247 */ 248 249 private void showInfoModal( com.sun.star.awt.XWindow xParent , 250 java.lang.String sTitle , 251 java.lang.String sMessage ) 252 { 253 try 254 { 255 // get access to the office toolkit environment 256 com.sun.star.awt.XToolkit xKit = (com.sun.star.awt.XToolkit)UnoRuntime.queryInterface( 257 com.sun.star.awt.XToolkit.class, 258 m_xCmpCtx.getServiceManager().createInstanceWithContext("com.sun.star.awt.Toolkit", 259 m_xCmpCtx)); 260 261 // describe the info box ini it's parameters 262 com.sun.star.awt.WindowDescriptor aDescriptor = new com.sun.star.awt.WindowDescriptor(); 263 aDescriptor.WindowServiceName = "infobox"; 264 aDescriptor.Bounds = new com.sun.star.awt.Rectangle(0,0,300,200); 265 aDescriptor.WindowAttributes = com.sun.star.awt.WindowAttribute.BORDER | 266 com.sun.star.awt.WindowAttribute.MOVEABLE | 267 com.sun.star.awt.WindowAttribute.CLOSEABLE; 268 aDescriptor.Type = com.sun.star.awt.WindowClass.MODALTOP; 269 aDescriptor.ParentIndex = 1; 270 aDescriptor.Parent = (com.sun.star.awt.XWindowPeer)UnoRuntime.queryInterface( 271 com.sun.star.awt.XWindowPeer.class, 272 xParent); 273 274 // create the info box window 275 com.sun.star.awt.XWindowPeer xPeer = xKit.createWindow(aDescriptor); 276 com.sun.star.awt.XMessageBox xInfoBox = (com.sun.star.awt.XMessageBox)UnoRuntime.queryInterface( 277 com.sun.star.awt.XMessageBox.class, 278 xPeer); 279 if (xInfoBox == null) 280 return; 281 282 // fill it with all given informations and show it 283 xInfoBox.setCaptionText(sTitle); 284 xInfoBox.setMessageText(sMessage); 285 xInfoBox.execute(); 286 } 287 catch(java.lang.Throwable exIgnore) 288 { 289 // ignore any problem, which can occur here. 290 // It's not really a bug for this example job, if 291 // it's message could not be printed out! 292 } 293 } 294 295 //___________________________________________ 296 297 private void showInfoNonModal( java.lang.String sTitle , 298 java.lang.String sMessage ) 299 { 300 // Couldnt be implemented really using the toolkit ... 301 // Because we need a parent anytime. 302 // And showing e.g. a java dialog can make some trouble 303 // inside office ... but we have no chance here. 304 final java.lang.String sFinalTitle = sTitle; 305 final java.lang.String sFinalMessage = sMessage; 306 307 // On Mac OS X, AWT/Swing must not be accessed from the AppKit thread, so call 308 // SwingUtilities.invokeLater always on a fresh thread to avoid that problem 309 // (also, the current thread must not wait for that fresh thread to terminate, 310 // as that would cause a deadlock if this thread is the AppKit thread): 311 final Runnable doRun = new Runnable() { 312 public void run() { 313 javax.swing.JOptionPane.showMessageDialog(null, sFinalMessage, sFinalTitle, javax.swing.JOptionPane.INFORMATION_MESSAGE); 314 } 315 }; 316 317 new Thread( doRun ) { 318 public void run() { javax.swing.SwingUtilities.invokeLater(doRun); } 319 }.start(); 320 } 321 322 //___________________________________________ 323 324 /** helper to print out the given argument list. 325 * 326 * @param lGenericConfig 327 * contains all shared configuration items for a job 328 * 329 * @param lJobConfig 330 * contains all job sepcific configuration items 331 * 332 * @param lEnvironment 333 * contains some environment informations 334 * 335 * @param lDynamicData 336 * contains optional data of a might corresponding dispatch() request 337 */ 338 339 private java.lang.String formatOutArgs(com.sun.star.beans.NamedValue[] lGenericConfig, 340 com.sun.star.beans.NamedValue[] lJobConfig , 341 com.sun.star.beans.NamedValue[] lEnvironment , 342 com.sun.star.beans.NamedValue[] lDynamicData ) 343 { 344 java.lang.StringBuffer sOut = new java.lang.StringBuffer(1024); 345 346 sOut.append("list \"Config\": "); 347 if (lGenericConfig==null) 348 sOut.append("0 items\n"); 349 else 350 { 351 int c = lGenericConfig.length; 352 sOut.append(c+" items\n"); 353 for (int i=0; i<c; ++i) 354 sOut.append("\t["+i+"] \""+lGenericConfig[i].Name+"\" = {"+lGenericConfig[i].Value+"}\n"); 355 } 356 sOut.append("list \"JobConfig\": "); 357 if (lJobConfig==null) 358 sOut.append("0 items\n"); 359 else 360 { 361 int c = lJobConfig.length; 362 sOut.append(c+" items\n"); 363 for (int i=0; i<c; ++i) 364 sOut.append("\t["+i+"] \""+lJobConfig[i].Name+"\" = {"+lJobConfig[i].Value+"}\n"); 365 } 366 sOut.append("list \"Environment\": "); 367 if (lEnvironment==null) 368 sOut.append("0 items\n"); 369 else 370 { 371 int c = lEnvironment.length; 372 sOut.append(c+" items\n"); 373 for (int i=0; i<c; ++i) 374 sOut.append("\t["+i+"] \""+lEnvironment[i].Name+"\" = {"+lEnvironment[i].Value+"}\n"); 375 } 376 sOut.append("list \"DynamicData\": "); 377 if (lDynamicData==null) 378 sOut.append("0 items\n"); 379 else 380 { 381 int c = lDynamicData.length; 382 sOut.append(c+" items\n"); 383 for (int i=0; i<c; ++i) 384 sOut.append("\t["+i+"] \""+lDynamicData[i].Name+"\" = {"+lDynamicData[i].Value+"}\n"); 385 } 386 387 return sOut.toString(); 388 } 389 390 public String[] getSupportedServiceNames() { 391 return SERVICENAMES; 392 } 393 394 public boolean supportsService( String sService ) { 395 int len = SERVICENAMES.length; 396 397 for( int i=0; i < len; i++) { 398 if ( sService.equals( SERVICENAMES[i] ) ) 399 return true; 400 } 401 402 return false; 403 } 404 405 public String getImplementationName() { 406 return( AsyncJob.class.getName() ); 407 } 408 409 410 //___________________________________________ 411 412 public synchronized static com.sun.star.lang.XSingleComponentFactory __getComponentFactory(java.lang.String sImplName) 413 { 414 com.sun.star.lang.XSingleComponentFactory xFactory = null; 415 if (sImplName.equals(AsyncJob.IMPLEMENTATIONNAME)) 416 xFactory = Factory.createComponentFactory(AsyncJob.class, SERVICENAMES); 417 418 return xFactory; 419 } 420 421 //___________________________________________ 422 // This method not longer necessary since OOo 3.4 where the component registration 423 // was changed to passive component registration. For more details see 424 // http://wiki.services.openoffice.org/wiki/Passive_Component_Registration 425 426 // public synchronized static boolean __writeRegistryServiceInfo(com.sun.star.registry.XRegistryKey xRegKey) 427 // { 428 // return Factory.writeRegistryServiceInfo( 429 // AsyncJob.IMPLEMENTATIONNAME, 430 // AsyncJob.SERVICENAMES, 431 // xRegKey); 432 // } 433 } 434