/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
*/
package util;
import com.sun.star.frame.XController;
import com.sun.star.frame.XDispatch;
import com.sun.star.frame.XDispatchProvider;
import com.sun.star.frame.XModel;
import com.sun.star.lang.XComponent;
import java.util.StringTokenizer;
import java.io.*;
import java.util.ArrayList;
import java.net.Socket;
import java.net.ServerSocket;
import java.net.URI;
import java.net.URISyntaxException;
import com.sun.star.awt.XToolkitExperimental;
import com.sun.star.beans.XPropertySet;
import com.sun.star.beans.Property;
import com.sun.star.lang.XMultiServiceFactory;
import com.sun.star.uno.UnoRuntime;
import com.sun.star.ucb.InteractiveAugmentedIOException;
import com.sun.star.ucb.XSimpleFileAccess;
import com.sun.star.lang.XServiceInfo;
import com.sun.star.util.URL;
import com.sun.star.util.XURLTransformer;
import com.sun.star.uno.AnyConverter;
import com.sun.star.uno.Type;
import com.sun.star.uno.XComponentContext;
import com.sun.star.util.XMacroExpander;
import java.text.DecimalFormat;
import java.util.Calendar;
import java.util.Collections;
import java.util.GregorianCalendar;
public class utils {
/**
*
* This method adds the DOCPTH to a given file
*
* @param sDocName the file which should be completed to the test doc path
* @return $TESTDOCPATH/sDocName
*/
public static String getFullTestDocName(String sDocName) {
String docpth = System.getProperty("DOCPTH");
if (docpth.endsWith("\\") || docpth.endsWith("/")) {
docpth = docpth.substring(0, docpth.length() - 1);
}
System.out.println("docpth:" + docpth);
String pthSep = System.getProperty("file.separator");
if (docpth.equals("unknown")) {
System.out.println("try to get tDoc from $SRC_ROOT/qadevOOo");
String srcRoot = System.getProperty(PropertyName.SRC_ROOT);
if (srcRoot != null) {
File srcR = new File(srcRoot);
String[] list = srcR.list(new FilenameFilter() {
public boolean accept(File dir, String name) {
return name.startsWith("qadevOOo");
}
});
if (list != null && list[0] != null) {
String tDoc = srcRoot.concat(pthSep).concat(list[0]).concat(pthSep).concat("testdocs");
if (new File(tDoc).exists()) {
docpth = tDoc;
}
}
}
}
if (docpth.startsWith("http:")) {
return docpth + "/" + sDocName;
}
String testdocPth = "";
if (docpth.equals("unknown")) {
System.out.println("try to get tDoc from OBJDSCS");
String objdscPth = System.getProperty("OBJDSCS");
if (objdscPth != null) {
int i = objdscPth.indexOf("objdsc");
String arcPth = objdscPth.substring(0, i - 1);
testdocPth = arcPth + pthSep + "doc" + pthSep + "java" +
pthSep + "testdocs" + pthSep + sDocName;
}
} else {
testdocPth = docpth + pthSep + sDocName;
}
return testdocPth;
}
/**
*
* This method adds the DOCPTH to a given file
* and changes the format to an file URL
*
*/
public static String getFullTestURL(String sDocName) {
String fullDocPath = getFullTestDocName(sDocName);
if (fullDocPath.startsWith("http:")) {
return fullDocPath;
}
if (fullDocPath.startsWith("file:")) {
return fullDocPath;
}
String prefix = null;
// Windows: \\\\margritte\\qaapi\\workspace\\qadev\\testdocs/emptyChart.sds
if (fullDocPath.startsWith("\\\\")) {
prefix = "file:";
}
fullDocPath = fullDocPath.replace('\\', '/');
if (prefix == null) {
if (fullDocPath.startsWith("//")) {
prefix = "file:/";
} else if (fullDocPath.startsWith("/")) {
prefix = "file://";
} else {
prefix = "file:///";
}
}
if (!fullDocPath.endsWith("/")) {
File aFile = new File(fullDocPath);
if (aFile.isDirectory()) {
fullDocPath += "/";
}
}
String fulldocURL = prefix + fullDocPath;
return fulldocURL;
}
/**
*
* This method changes a given URL to a valid file URL
*
*/
public static String getFullURL(String sDocName) {
String fullDocPath = sDocName;
fullDocPath = fullDocPath.replace('\\', '/');
if (fullDocPath.startsWith("http:")) {
return fullDocPath;
}
if (fullDocPath.startsWith("ftp:")) {
return fullDocPath;
}
String prefix = "";
if (!fullDocPath.startsWith("file:///")) {
if (fullDocPath.startsWith("//")) {
prefix = "file:";
} else {
if (fullDocPath.startsWith("/")) {
prefix = "file://";
}
else
{
prefix = "file:///";
}
}
}
if (!fullDocPath.endsWith("/")) {
File aFile = new File(fullDocPath);
if (aFile.isDirectory()) {
fullDocPath += "/";
}
}
String fulldocURL = prefix + fullDocPath;
return fulldocURL;
}
/**
*
* This method gets the user dir of the connected office
*
*/
public static String getOfficeUserPath(XMultiServiceFactory msf) {
// get a folder located in the user dir
String userPath = getOfficeSettingsValue(msf, "UserConfig");
// strip the returned folder to the user dir
if (userPath.charAt(userPath.length() - 1) == '/') {
userPath = userPath.substring(0, userPath.length() - 1);
}
int index = userPath.lastIndexOf('/');
if (index != -1) {
userPath = userPath.substring(0, index);
}
return userPath;
}
/**
* In the office there are some settings available. This function
* returns the value of the given setting name. For Example the setting name "Temp"
* "Temp" returns the temp folder of the office instance.
* @param msf a XMultiServiceFactory
* @param setting the name of the setting the value should be returned.
* For example "Temp" returns the temp folder of the current office instance.
* @see com.sun.star.util.PathSettings
* @return the value as String
*/
private static String getOfficeSettingsValue(XMultiServiceFactory msf, String setting) {
try {
Object settings = msf.createInstance("com.sun.star.comp.framework.PathSettings");
XPropertySet pthSettings = (XPropertySet) AnyConverter.toObject(
new Type(XPropertySet.class), settings);
return (String) pthSettings.getPropertyValue(setting);
} catch (com.sun.star.uno.Exception ex) {
throw new RuntimeException(ex);
}
}
/**
* This method returns the temp directory of the user.
* Since Java 1.4 it is not possible to read environment variables. To workaround
* this, the Java parameter -D could be used.
*/
public static String getUsersTempDir() {
String tempDir = System.getProperty("my.temp");
if (tempDir == null) {
tempDir = System.getProperty("my.tmp");
if (tempDir == null) {
tempDir = System.getProperty("java.io.tmpdir");
}
}
// remove ending file separator
if (tempDir.endsWith(System.getProperty("file.separator"))) {
tempDir = tempDir.substring(0, tempDir.length() - 1);
}
return tempDir;
}
/**
*
* This method gets the temp dir of the connected office
*
*/
public static String getOfficeTemp(XMultiServiceFactory msf) {
String url = getOfficeUserPath(msf) + "/test-temp/";
try {
new File(new URI(url)).mkdir();
} catch (URISyntaxException e) {
throw new RuntimeException(e);
}
return url;
}
/**
* Gets StarOffice temp directory without 'file:///' prefix.
* For example is useful for Registry URL specifying.
* @msf Office factory for accessing its settings.
* @return SOffice temporary directory in form for example
* 'd:/Office60/user/temp/'.
*/
public static String getOfficeTempDir(XMultiServiceFactory msf) {
String dir = getOfficeTemp(msf);
int idx = dir.indexOf("file:///");
if (idx < 0) {
return dir;
}
dir = dir.substring("file:///".length());
idx = dir.indexOf(':');
// is the last character a '/' or a '\'?
boolean lastCharSet = dir.endsWith("/") || dir.endsWith("\\");
if (idx < 0) { // linux or solaris
dir = "/" + dir;
dir += lastCharSet ? "" : "/";
} else { // windows
dir += lastCharSet ? "" : "\\";
}
return dir;
}
/**
* Gets StarOffice temp directory without 'file:///' prefix.
* and System dependent file separator
*/
public static String getOfficeTempDirSys(XMultiServiceFactory msf) {
String dir = getOfficeTemp(msf);
String sysDir = "";
int idx = dir.indexOf("file://");
// remove leading 'file://'
if (idx < 0) {
sysDir = dir;
} else {
sysDir = dir.substring("file://".length());
}
// append '/' if not there (e.g. linux)
if (sysDir.charAt(sysDir.length() - 1) != '/') {
sysDir += "/";
}
// remove leading '/' and replace others with '\' on windows machines
if (sysDir.indexOf(':') != -1) {
sysDir = sysDir.substring(1);
sysDir = sysDir.replace('/', '\\');
}
return sysDir;
}
/**
* converts a fileURL to a system URL
* @param fileURL a file URL
* @return a system URL
*/
public static String getSystemURL(String fileURL) {
String sysDir = "";
int idx = fileURL.indexOf("file://");
// remove leading 'file://'
if (idx < 0) {
sysDir = fileURL;
} else {
sysDir = fileURL.substring("file://".length());
}
// remove leading '/' and replace others with '\' on windows machines
if (sysDir.indexOf(':') != -1) {
sysDir = sysDir.substring(1);
sysDir = sysDir.replace('/', '\\');
}
return sysDir;
}
/**
* This method check via Office the existence of the given file URL
* @param msf the multiservice factory
* @param fileURL the file which existence should be checked
* @return true if the file exists, else false
*/
public static boolean fileExists(XMultiServiceFactory msf, String fileURL) throws com.sun.star.uno.Exception {
Object fileacc = msf.createInstance("com.sun.star.comp.ucb.SimpleFileAccess");
XSimpleFileAccess simpleAccess = UnoRuntime.queryInterface(XSimpleFileAccess.class,
fileacc);
return simpleAccess.exists(fileURL);
}
/**
* This method deletes via office the given file URL. It checks the existence
* of fileURL
. If exists it will be deleted.
* @param xMsf the multiservice factory
* @param fileURL the file to delete
* @return true if the file could be deleted or the file does not exist
*/
public static boolean deleteFile(XMultiServiceFactory xMsf, String fileURL) {
boolean delete = true;
try {
Object fileacc = xMsf.createInstance("com.sun.star.comp.ucb.SimpleFileAccess");
XSimpleFileAccess simpleAccess = UnoRuntime.queryInterface(XSimpleFileAccess.class,
fileacc);
if (simpleAccess.exists(fileURL)) {
simpleAccess.kill(fileURL);
}
} catch (Exception e) {
System.out.println("Couldn't delete file '" + fileURL + "'");
e.printStackTrace();
delete = false;
}
return delete;
}
/**
* This method copies via office a given file to a new one
* @param xMsf the multi service factory
* @param source the source file
* @param destination the destination file
* @return true at success
*/
public static boolean copyFile(XMultiServiceFactory xMsf, String source, String destination) {
boolean res = false;
try {
Object fileacc = xMsf.createInstance("com.sun.star.comp.ucb.SimpleFileAccess");
XSimpleFileAccess simpleAccess = UnoRuntime.queryInterface(XSimpleFileAccess.class,
fileacc);
if (!simpleAccess.exists(destination)) {
simpleAccess.copy(source, destination);
}
res = true;
} catch (Exception e) {
System.out.println("Couldn't copy file '" + source + "' -> '" + destination + "'");
e.printStackTrace();
res = false;
}
return res;
}
private static void overwriteFile_impl(
XMultiServiceFactory xMsf, String oldF, String newF)
throws InteractiveAugmentedIOException
{
try {
Object fileacc = xMsf.createInstance("com.sun.star.comp.ucb.SimpleFileAccess");
XSimpleFileAccess simpleAccess = UnoRuntime.queryInterface(XSimpleFileAccess.class,
fileacc);
if (simpleAccess.exists(newF)) {
simpleAccess.kill(newF);
}
simpleAccess.copy(oldF, newF);
} catch (InteractiveAugmentedIOException e) {
throw e;
} catch (com.sun.star.uno.Exception ex) {
throw new RuntimeException("Could not copy " + oldF + " to " + newF, ex);
}
}
/**
* Copies file to a new location using OpenOffice.org features. If the target
* file already exists, the file is deleted.
*
* @returns true
if the file was successfully copied,
* false
if some errors occurred (e.g. file is locked, used
* by another process).
*/
public static boolean tryOverwriteFile(
XMultiServiceFactory xMsf, String oldF, String newF)
{
try {
overwriteFile_impl(xMsf, oldF, newF);
} catch (InteractiveAugmentedIOException e) {
return false;
}
return true;
}
public static boolean hasPropertyByName(XPropertySet props, String aName) {
Property[] list = props.getPropertySetInfo().getProperties();
boolean res = false;
for (int i = 0; i < list.length; i++) {
String the_name = list[i].Name;
if (aName.equals(the_name)) {
res = true;
}
}
return res;
}
/**
*
* This method returns the implementation name of a given object
*
*/
public static String getImplName(Object aObject) {
XServiceInfo xSI = UnoRuntime.queryInterface(XServiceInfo.class, aObject);
return xSI == null ? "Unknown, does not implement XServiceInfo" : xSI.getImplementationName();
}
/**
*
* This method checks if an Object is void
*
*/
public static boolean isVoid(Object aObject) {
if (aObject instanceof com.sun.star.uno.Any) {
com.sun.star.uno.Any oAny = (com.sun.star.uno.Any) aObject;
return oAny.getType().getTypeName().equals("void");
} else {
return false;
}
}
/**
* Scan localhost for the next free port-number from a starting port
* on. If the starting port is smaller than 1024, port number starts with
* 10000 as default, because numbers < 1024 are never free on unix machines.
* @param startPort The port where scanning starts.
* @return The next free port.
*/
public static int getNextFreePort(int startPort) {
if (startPort < 1024) {
startPort = 10000;
}
for (int port = startPort; port < 65536; port++) {
System.out.println("Scan port " + port);
try {
// first trying to establish a server-socket on localhost
// fails if there is already a server running
ServerSocket sSock = new ServerSocket(port);
sSock.close();
} catch (IOException e) {
System.out.println(" -> server: occupied port " + port);
continue;
}
try {
Socket sock = new Socket("localhost", port);
System.out.println(" -> socket: occupied port: " + port);
try {
sock.close();
} catch (IOException ex) {
// ignore close exception
}
} catch (IOException e) {
System.out.println(" -> free port");
return port;
}
}
return 65535;
}
public static URL parseURL(XMultiServiceFactory xMSF, String url) {
URL[] rUrl = new URL[1];
rUrl[0] = new URL();
rUrl[0].Complete = url;
XURLTransformer xTrans = null;
try {
Object inst = xMSF.createInstance("com.sun.star.util.URLTransformer");
xTrans = UnoRuntime.queryInterface(XURLTransformer.class, inst);
} catch (com.sun.star.uno.Exception e) {
}
if (xTrans != null)
xTrans.parseStrict(rUrl);
return rUrl[0];
}
public static String getOfficeURL(XMultiServiceFactory msf) throws com.sun.star.uno.Exception {
Object settings = msf.createInstance("com.sun.star.util.PathSettings");
XPropertySet settingProps = UnoRuntime.queryInterface(XPropertySet.class, settings);
String path = (String) settingProps.getPropertyValue("Module");
return path;
}
/**
* Get an array of all property names from the property set. With the include
* and exclude parameters the properties can be filtered.
* Set excludePropertyAttribute = 0 and includePropertyAttribute = 0
* to include all and exclude none.
* @param props The instance of XPropertySet
* @param includePropertyAttribute Properties without these attributes are filtered and will not be returned.
* @param excludePropertyAttribute Properties with these attributes are filtered and will not be returned.
* @return A String array with all property names.
* @see com.sun.star.beans.XPropertySet
* @see com.sun.star.beans.Property
* @see com.sun.star.beans.PropertyAttribute
*/
public static String[] getFilteredPropertyNames(XPropertySet props, short includePropertyAttribute,
short excludePropertyAttribute) {
Property[] the_props = props.getPropertySetInfo().getProperties();
ArrayList l = new ArrayList();
for (int i = 0; i < the_props.length; i++) {
boolean exclude = ((the_props[i].Attributes & excludePropertyAttribute) != 0);
boolean include = (includePropertyAttribute == 0) ||
((the_props[i].Attributes & includePropertyAttribute) != 0);
if (include && !exclude) {
l.add(the_props[i].Name);
}
}
Collections.sort(l);
String[] names = new String[l.size()];
names = l.toArray(names);
return names;
}
/** Causes the thread to sleep some time.
* This is the default call, which waits for 500ms.
*/
public static void shortWait() {
pause(utils.DEFAULT_SHORT_WAIT_MS);
}
/** Causes the thread to sleep some time.
*/
public static void pause(int milliseconds) {
try {
Thread.sleep(milliseconds);
} catch (InterruptedException e) {
System.out.println("While waiting :" + e);
}
}
public static void waitForEventIdle(XMultiServiceFactory xMSF) {
try {
XToolkitExperimental xToolkit = UnoRuntime.queryInterface(
XToolkitExperimental.class,
xMSF.createInstance("com.sun.star.awt.Toolkit"));
xToolkit.processEventsToIdle();
} catch (com.sun.star.uno.Exception ex) {
throw new RuntimeException(ex);
}
}
/**
* Validate the AppExecutionCommand. Returned is an error message, starting
* with "Error:", or a warning, if the command might work.
* @param appExecCommand The application execution command that is checked.
* @param os The operating system where the check runs.
* @return The error message, or OK, if no error was detected.
*/
public static String validateAppExecutionCommand(String appExecCommand, String os) {
String errorMessage = "OK";
appExecCommand = appExecCommand.replace("\"", "");
appExecCommand = appExecCommand.replace("'", "");
StringTokenizer commandTokens = new StringTokenizer(appExecCommand, " \t");
String officeExecCommand = "soffice";
StringBuilder sb = new StringBuilder();
// is there a 'soffice' in the command? 2do: eliminate case sensitivity on windows
int index = -1;
while (commandTokens.hasMoreTokens() && index == -1) {
sb.append(commandTokens.nextToken()).append(" ");
index = sb.indexOf(officeExecCommand);
}
if (index == -1) {
errorMessage = "Error: Your 'AppExecutionCommand' parameter does not " +
"contain '" + officeExecCommand + "'.";
} else {
String officeExecutable = sb.toString();
// does the directory exist?
officeExecutable = officeExecutable.trim();
String officePath = officeExecutable.substring(0, index);
File f = new File(officePath);
if (!f.exists() || !f.isDirectory()) {
errorMessage = "Error: Your 'AppExecutionCommand' parameter does not " +
"point to a valid system directory: " + officePath;
} else {
// is it an office installation?
f = new File(officeExecutable);
// one try for windows platform can't be wrong...
if (!f.exists() || !f.isFile()) {
f = new File(officeExecutable + ".exe");
}
if (!f.exists() || !f.isFile()) {
errorMessage = "Error: Your 'AppExecutionCommand' parameter does not " +
"point to a valid office installation.";
} else {
// do we have the accept parameter?
boolean gotNoAccept = true;
while (commandTokens.hasMoreElements()) {
String officeParam = commandTokens.nextToken();
if (officeParam.indexOf("--accept=") != -1) {
gotNoAccept = false;
errorMessage = validateConnectString(officeParam, true);
}
}
if (gotNoAccept) {
errorMessage = "Error: Your 'AppExecutionCommand' parameter does not " +
"contain a '--accept' parameter for connecting the office.";
}
}
}
}
return errorMessage;
}
/**
* Validate the connection string. Returned is an error message, starting
* with "Error:", or a warning, if the command might work.
* @param connectString The connection string that is checked.
* @param checkAppExecutionCommand If the AppExecutionCommand is checked, the error message is different.
* @return The error message, or OK, if no error was detected.
*/
private static String validateConnectString(String connectString, boolean checkAppExecutionCommand) {
String acceptPrefix = "";
if (checkAppExecutionCommand) {
acceptPrefix = "--accept=";
}
String errorMessage = "OK";
// a warning, if an unknown connection method is used
if (connectString.indexOf("socket") != -1) {
if (connectString.indexOf(acceptPrefix + "socket,host=") == -1 ||
connectString.indexOf("port=") == -1) {
if (checkAppExecutionCommand) {
errorMessage = "Error: The '--accept' parameter contains a syntax error: It should be like: '--accept=socket,host=localhost,port=8100;urp;";
} else {
errorMessage = "Error: The 'ConnectionString' parameter contains a syntax error: It should be like: 'socket,host=localhost,port=8100'";
}
}
} else if (connectString.indexOf("pipe") != -1) {
if (connectString.indexOf(acceptPrefix + "pipe,name=") == -1) {
if (checkAppExecutionCommand) {
errorMessage = "Error: The '--accept' parameter contains a syntax error: It should be like: '--accept=pipe,name=myuniquename;urp;'";
} else {
errorMessage = "Error: The 'ConnectionString' parameter contains a syntax error: It should be like: 'pipe,name=myuniquename'";
}
}
} else {
if (checkAppExecutionCommand) {
errorMessage = "Warning: The '--accept' parameter contains an unknown connection method.";
} else {
errorMessage = "Warning: The 'ConnectionString' parameter contains an unknown connection method.";
}
}
return errorMessage;
}
/**
* expand macrofied strings like ${$ORIGIN/bootstrap.ini:UserInstallation}
or
* $_OS
* @param xMSF the MultiServiceFactory
* @param expand the string to expand
* @throws java.lang.Exception was thrown on any exception
* @return return the expanded string
* @see com.sun.star.util.XMacroExpander
*/
public static String expandMacro(XMultiServiceFactory xMSF, String expand) {
try {
XPropertySet xPS = UnoRuntime.queryInterface(XPropertySet.class, xMSF);
XComponentContext xContext = UnoRuntime.queryInterface(XComponentContext.class,
xPS.getPropertyValue("DefaultContext"));
XMacroExpander xME = UnoRuntime.queryInterface(XMacroExpander.class,
xContext.getValueByName("/singletons/com.sun.star.util.theMacroExpander"));
return xME.expandMacros(expand);
} catch (Exception e) {
throw new RuntimeException("could not expand macro", e);
}
}
/**
* dispatches given URL
to the document XComponent
* @param xMSF the XMultiServiceFactory
* @param xDoc the document where to dispatch
* @param URL the URL
to dispatch
* @throws java.lang.Exception throws java.lang.Exception
on any error
*/
public static void dispatchURL(XMultiServiceFactory xMSF, XComponent xDoc, String URL) throws java.lang.Exception {
XModel aModel = UnoRuntime.queryInterface(XModel.class, xDoc);
XController xCont = aModel.getCurrentController();
dispatchURL(xMSF, xCont, URL);
}
/**
* dispatches given URL
to the XController
* @param xMSF the XMultiServiceFactory
* @param xCont the XController
to query for a XDispatchProvider
* @param URL the URL
to dispatch
*/
private static void dispatchURL(XMultiServiceFactory xMSF, XController xCont, String URL) {
try {
XDispatchProvider xDispProv = UnoRuntime.queryInterface(XDispatchProvider.class, xCont);
XURLTransformer xParser = UnoRuntime.queryInterface(
XURLTransformer.class,
xMSF.createInstance("com.sun.star.util.URLTransformer"));
// Because it's an in/out parameter we must use an array of URL objects.
URL[] aParseURL = new URL[1];
aParseURL[0] = new URL();
aParseURL[0].Complete = URL;
xParser.parseStrict(aParseURL);
URL aURL = aParseURL[0];
XDispatch xDispatcher = xDispProv.queryDispatch(aURL, "", 0);
xDispatcher.dispatch(aURL, null);
waitForEventIdle(xMSF);
} catch (Exception e) {
throw new RuntimeException("Could not dispatch URL '" + URL + "'", e);
}
}
/** returns a String which contains the current date and time
* format: [DD.MM.YYYY - HH:MM:SS::mm]
*
** @return a String which contains the current date and time
*/
public static String getDateTime() {
Calendar cal = new GregorianCalendar();
DecimalFormat dfmt = new DecimalFormat("00");
String dateTime = dfmt.format(cal.get(Calendar.DAY_OF_MONTH)) + "." +
dfmt.format(cal.get(Calendar.MONTH) + 1) + "." +
dfmt.format(cal.get(Calendar.YEAR)) + " - " +
dfmt.format(cal.get(Calendar.HOUR_OF_DAY)) + ":" +
dfmt.format(cal.get(Calendar.MINUTE)) + ":" +
dfmt.format(cal.get(Calendar.SECOND)) + "," +
dfmt.format(cal.get(Calendar.MILLISECOND));
return "[" + dateTime + "]";
}
/**
* Default short wait time for the Office
*/
public static final int DEFAULT_SHORT_WAIT_MS = 500;
/// see rtl/math.hxx
public static boolean approxEqual( double a, double b )
{
if( a == b )
return true;
double x = a - b;
return (x < 0.0 ? -x : x)
< ((a < 0.0 ? -a : a) * (1.0 / (16777216.0 * 16777216.0)));
}
}