src.com.ibm.as400.util.servlet.HTMLTableConverter Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jt400-jdk9 Show documentation
Show all versions of jt400-jdk9 Show documentation
The Open Source version of the IBM Toolbox for Java
The newest version!
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename: HTMLTableConverter.java
//
// The source code contained herein is licensed under the IBM Public License
// Version 1.0, which has been approved by the Open Source Initiative.
// Copyright (C) 1997-2000 International Business Machines Corporation and
// others. All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////
package com.ibm.as400.util.servlet;
import com.ibm.as400.util.html.HTMLHyperlink;
import com.ibm.as400.util.html.HTMLTable;
import com.ibm.as400.util.html.HTMLTableCell;
import com.ibm.as400.util.html.HTMLTableHeader;
import com.ibm.as400.util.html.HTMLTableRow;
import com.ibm.as400.util.html.HTMLTagElement;
import com.ibm.as400.util.html.HTMLText;
import com.ibm.as400.util.html.LineLayoutFormPanel; // @D4A
import com.ibm.as400.access.ActionCompletedEvent;
import com.ibm.as400.access.ActionCompletedListener;
import com.ibm.as400.access.ExtendedIllegalArgumentException;
import com.ibm.as400.access.ExtendedIllegalStateException;
import com.ibm.as400.access.Trace;
import java.beans.PropertyChangeSupport;
import java.beans.PropertyChangeListener;
import java.beans.PropertyVetoException;
import java.beans.VetoableChangeSupport;
import java.beans.VetoableChangeListener;
import java.io.Serializable;
import java.util.Vector;
/**
* The HTMLTableConverter class can be used to convert the data from a RowData
* object to a single HTML table for a selected group or page of row data, an
* array of strings or HTML tables. Each resulting HTML table then can be used
* by a servlet to display the rowdata to a browser.
*
* HTMLTableConverter objects generate the following events:
*
* - ActionCompletedEvent
* - SectionCompletedEvent
* - PropertyChangeEvent
* - VetoableChangeEvent
*
*
* The following example creates an HTMLTableConverter object and does the conversion.
*
* // Create an HTMLTableConverter object.
* HTMLTableConverter converter = new HTMLTableConverter();
* // Setup the table tag with a maximum of 25 rows/table.
* HTMLTable table = new HTMLTable();
* converter.setMaximumTableSize(25);
* converter.setTable(table);
* // Convert the row data.
* // Assume the RowData object was created and initialized in a previous step.
* String[] html = converter.convert(rowdata);
*
**/
public class HTMLTableConverter extends StringConverter implements Serializable
{
static final long serialVersionUID = 9154342923705960360L;
private HTMLTable htmlTable_; // The html table.
private HTMLHyperlink[] links_; // The table column header hyperlinks.
private int maxTableSize_ = 0; // The maximum number of rows in a table.
private boolean useMetaData_ = false; // Whether the metadata is used to create the table header. Otherwise, the existing table header is used.
transient private Vector completedListeners_; // The conversion completed listeners.
transient private PropertyChangeSupport changes_; // The property change listeners.
transient private VetoableChangeSupport vetos_; // The vetoable change listeners.
transient private SectionCompletedSupport sectionCompletedSupport_; // The section completed listeners.
/**
* Constructs a default HTMLTableConverter object.
**/
public HTMLTableConverter()
{
super();
// Initialize the transient data (listeners).
// initializeTransient(); @CRS
}
/**
* Adds an ActionCompletedListener.
* The specified ActionCompletedListener's actionCompleted method is called
* each time the table conversion is complete and all the row data is converted.
* The ActionCompletedListener object is added to a list of ActionCompletedListeners
* managed by this class; it can be removed with removeActionCompletedListener.
*
* @param listener The ActionCompletedListener.
* @see #removeActionCompletedListener
**/
public void addActionCompletedListener(ActionCompletedListener listener)
{
if (listener == null)
throw new NullPointerException("listener");
if (completedListeners_ == null) completedListeners_ = new Vector(); //@CRS
completedListeners_.addElement(listener);
}
/**
* Adds a PropertyChangeListener. The specified PropertyChangeListener's propertyChange
* method is called each time the value of any bound property is changed.
* @param listener The PropertyChangeListener.
* @see #removePropertyChangeListener
**/
public void addPropertyChangeListener(PropertyChangeListener listener)
{
if (listener == null)
throw new NullPointerException("listener");
if (changes_ == null) changes_ = new PropertyChangeSupport(this); //@CRS
changes_.addPropertyChangeListener(listener);
}
/**
* Adds a SectionCompletedListener.
* The specified SectionCompletedListener's sectionCompleted method is called
* each time the conversion for a single table is complete.
* The SectionCompletedListener object is added to an internal list of SectionCompletedListeners;
* it can be removed with removeSectionCompletedListener.
*
* @param listener The SectionCompletedListener.
* @see #removeSectionCompletedListener
**/
public void addSectionCompletedListener(SectionCompletedListener listener)
{
if (sectionCompletedSupport_ == null) sectionCompletedSupport_ = new SectionCompletedSupport(this); //@CRS
sectionCompletedSupport_.addSectionCompletedListener(listener);
}
/**
* Adds the VetoableChangeListener. The specified VetoableChangeListener's vetoableChange
* method is called each time the value of any constrained property is changed.
* @param listener The VetoableChangeListener.
* @see #removeVetoableChangeListener
**/
public void addVetoableChangeListener(VetoableChangeListener listener)
{
if (listener == null)
throw new NullPointerException("listener");
if (vetos_ == null) vetos_ = new VetoableChangeSupport(this); //@CRS
vetos_.addVetoableChangeListener(listener);
}
/**
* Calculates the number of tables in the array.
* @param numberRows The number of row in the list.
* @return The array of HTMLTables objects.
**/
private HTMLTable[] calculateNumberOfTables(int numberRows)
{
int numTables = 1;
if (maxTableSize_ > 0 && numberRows > 0) // @C1
{
numTables = numberRows / maxTableSize_;
if (numberRows % maxTableSize_ != 0)
numTables++;
}
return new HTMLTable[numTables];
}
/**
* Converts the row data specified by rowdata into an array of HTMLTable objects.
* If the default table has not been set, it is automatically created with the
* column header information being obtained from the metadata.
* @param rowdata The RowData object that contains the row data.
* @param metadata The RowMetaData object that contains the metadata.
* @return An array of HTMLTables.
* @exception PropertyVetoException If a change is vetoed.
* @exception RowDataException If a row data error occurs.
* @see #setTable
**/
private HTMLTable[] convertRowData(RowData rowdata, RowMetaData metadata) //@A1 //$D2C
throws PropertyVetoException, RowDataException
{
HTMLTable[] tables = calculateNumberOfTables(rowdata.length());
for (int i = 0; i < tables.length; ++i)
{
tables[i] = convertRowData(rowdata, metadata, i);
}
return tables;
}
/**
* Converts the row data specified by rowdata at a specfic page into an HTMLTable object.
* If the default table has not been set, it is automatically created with the
* column header information being obtained from the metadata.
* @param rowdata The RowData object that contains the row data.
* @param metadata The RowMetaData object that contains the metadata.
* @param page A specific page of the row data.
* @return An HTMLTable.
* @exception PropertyVetoException If a change is vetoed.
* @exception RowDataException If a row data error occurs.
* @see #setTable
**/
private HTMLTable convertRowData(RowData rowdata, RowMetaData metadata, int page) //$D2A
throws PropertyVetoException, RowDataException
{
if (metadata == null)
{
Trace.log(Trace.ERROR, "The rowdata's metadata attribute is invalid.");
throw new ExtendedIllegalStateException("rowdata metadata", ExtendedIllegalStateException.PROPERTY_NOT_SET);
}
// Validate the page parameter.
if (page < 0)
throw new ExtendedIllegalArgumentException("page", ExtendedIllegalArgumentException.RANGE_NOT_VALID);
// Create the table to be used.
if (htmlTable_ == null)
{
// Use the metadata for the column headers.
setUseMetaData(true);
try
{
htmlTable_ = new HTMLTable();
htmlTable_.setHeaderInUse(false);
}
catch (PropertyVetoException veto)
{ /* will never occur. */
}
}
// Set the table header based on the metadata.
if (isUseMetaData())
setTableHeader(metadata);
// Create and initialize the Table.
HTMLTable table = createDefaultTable();
// If there is no rowdata, then return an empty table. //$D3A
if (rowdata.length() == 0) //$D3A
return table; //$D3A
// Row processing variables.
long numRowsInTable = 1;
int numColumns = metadata.getColumnCount();
// Process the row data.
if (page == 0)
rowdata.beforeFirst();
else
rowdata.absolute((maxTableSize_ * page)-1);
// Keep track of which row we are at in the table.
int rowLocation = 0;
// If no max table size is set, then the max will be the
// size of the row data.
if (maxTableSize_ == 0)
maxTableSize_ = rowdata.length();
while (rowdata.next() && (maxTableSize_ > rowLocation))
{
// Determine if the table is at the maximum size.
if (maxTableSize_ > 0)
{
if (numRowsInTable > 1 && (numRowsInTable % maxTableSize_ == 1) )
{
// Notify the listeners that a table is finished.
if (sectionCompletedSupport_ != null) sectionCompletedSupport_.fireSectionCompleted(table.getTag()); //@CRS
}
}
// Start the row (default row from table).
HTMLTableRow row = new HTMLTableRow();
Vector properties;
for (int column=0; column< numColumns; column++)
{
// Create a default cell.
HTMLTableCell cell = new HTMLTableCell();
if (metadata.isNumericData(column) == true) // @C1
cell.setHorizontalAlignment(HTMLTableCell.RIGHT);
HTMLTagElement element;
// Check object properties for a specific table cell to use.
properties = rowdata.getObjectProperties(column);
if (properties != null)
{
int propSize = properties.size();
for (int index=0; index< propSize; index++)
{
// Use the local cell tag.
if (properties.elementAt(index) instanceof HTMLTableCell)
cell = (HTMLTableCell)properties.elementAt(index);
}
}
// Set the column data.
Object columnObject = rowdata.getObject(column);
// If the column data is null, place a
into the cell otherwise // @D4A
// a NullPointerException will be thrown for an empty cell elment. // @D4A
if (columnObject == null) // @D4A
columnObject = new LineLayoutFormPanel(); // @D4A
try
{
cell.setElement((HTMLTagElement)columnObject);
}
catch (ClassCastException e)
{
cell.setElement(new HTMLText(columnObject.toString()));
}
if (metadata.getColumnAlignment(column) != null) //@D5A
cell.setHorizontalAlignment(metadata.getColumnAlignment(column)); //@D5A
if (metadata.getColumnDirection(column) != null) //@D5A
cell.setDirection(metadata.getColumnDirection(column)); //@D5A
// Add the column cell to the row.
row.addColumn(cell);
}
// Add the row of data to the table.
table.addRow(row);
numRowsInTable++;
rowLocation++;
}
// Notify section completed listeners that the last table is converted.
if (sectionCompletedSupport_ != null) sectionCompletedSupport_.fireSectionCompleted(table.getTag()); //@CRS
// Notify listeners that the tables have been converted.
fireCompleted();
return table;
}
/**
* Converts the row data specified by rowdata into an array of HTMLTable objects.
* @param rowdata The RowData object that contains the row data.
* @return An array of HTMLTable objects.
* @exception PropertyVetoException If a change is vetoed.
* @exception RowDataException If a row data error occurs.
**/
public HTMLTable[] convertToTables(RowData rowdata) throws PropertyVetoException, RowDataException // @A1 $D2C
{
if (rowdata == null)
throw new NullPointerException("rowdata");
HTMLTable[] tables = convertRowData(rowdata, rowdata.getMetaData());
//Return the list of HTML tables.
return tables;
}
/**
* Converts the row data specified by rowdata at the specified page into an HTMLTable object
* when using the maximum table size.
* @param rowdata The RowData object that contains the row data.
* @param page The specific page of the row data.
* @return An HTMLTable object.
* @exception PropertyVetoException If a change is vetoed.
* @exception RowDataException If a row data error occurs.
**/
public HTMLTable convertToTable(RowData rowdata, int page) throws PropertyVetoException, RowDataException //$D2A
{
if (rowdata == null)
throw new NullPointerException("rowdata");
HTMLTable table = convertRowData(rowdata, rowdata.getMetaData(), page);
// Return the HTML table.
return table;
}
/**
* Creates a default HTMLTable.
* @return An HTMLTable object.
**/
private HTMLTable createDefaultTable()
{
HTMLTable table = new HTMLTable();
try
{
if (htmlTable_.getHeader() != null)
table.setHeader(htmlTable_.getHeader()); // header
if (htmlTable_.getCaption() != null)
table.setCaption(htmlTable_.getCaption()); // caption
if (htmlTable_.getAlignment() != null)
table.setAlignment(htmlTable_.getAlignment()); // alignment
table.setBorderWidth(htmlTable_.getBorderWidth()); // border width
table.setCellPadding(htmlTable_.getCellPadding()); // cell padding
table.setCellSpacing(htmlTable_.getCellSpacing()); // cell spacing
table.setWidth(htmlTable_.getWidth(), htmlTable_.isWidthInPercent()); // width
table.setHeaderInUse(htmlTable_.isHeaderInUse()); // header usage
}
catch (PropertyVetoException veto)
{ /* will never occur. */
}
return table;
}
/**
* Converts the row data to a String array of HTML tables.
* If the default table has not been set, it is automatically created with the
* column header information being obtained from the metadata.
* @param rowdata The RowData object that contains the row data.
* @param metadata The RowMetaData object that contains the metadata.
*
* @return An array of HTML strings.
* @exception PropertyVetoException If a change is vetoed.
* @exception RowDataException If a row data error occurs.
* @see #setTable
**/
String[] doConvert(RowData rowdata, RowMetaData metadata) throws PropertyVetoException, RowDataException
{
HTMLTable[] tables = convertRowData(rowdata, metadata); //$D2C
// Return the list of tables as String array.
String[] data = new String[tables.length];
for (int i=0; i< data.length; i++)
data[i] = tables[i].getTag();
return data;
}
/**
* Fires a completed event to notify that all the tables have been converted.
**/
private void fireCompleted()
{
if (completedListeners_ == null) return; //@CRS
Vector targets = (Vector) completedListeners_.clone();
ActionCompletedEvent event = new ActionCompletedEvent(this);
for (int i=0; i< targets.size(); i++)
{
ActionCompletedListener target = (ActionCompletedListener)targets.elementAt(i);
target.actionCompleted(event);
}
}
/**
* Returns the table header's hyperlinks.
* @return The hyperlinks.
**/
public HTMLHyperlink[] getHeaderHyperlinks()
{
return links_;
}
/**
* Returns the maximum number of rows in a table.
* The default maximum size is 0 (no maximum).
* @return The maximum size.
**/
public int getMaximumTableSize()
{
return maxTableSize_;
}
/**
* Returns the object hyperlink for the current row's specified column.
* @param rowdata The RowData object that contains the data.
* @param column The column number (0-based).
*
* @return The hyperlink.
**/
public HTMLHyperlink getObjectHyperlink(RowData rowdata, int column)
{
// Validate the rowdata parameter.
if (rowdata == null)
throw new NullPointerException("rowdata");
return getObjectHyperlink(rowdata, rowdata.getCurrentPosition(), column);
}
/**
* Returns the object's hyperlink at the specified row and column.
* @param rowdata The RowData object that contains the data.
* @param row The row number (0-based).
* @param column The column number (0-based).
*
* @return The hyperlink.
**/
public HTMLHyperlink getObjectHyperlink(RowData rowdata, int row, int column)
{
// Validate the rowdata parameter.
if (rowdata == null)
throw new NullPointerException("rowdata");
// Position to the row.
if (!rowdata.absolute(row)) // Validates the row parameter.
throw new ExtendedIllegalArgumentException("row", ExtendedIllegalArgumentException.RANGE_NOT_VALID);
// Get the object's properties.
Vector properties = rowdata.getObjectProperties(column); // Validates the column parameter.
HTMLHyperlink link = null;
if (properties != null)
{
// Get the hyperlink associated with the object.
int size = properties.size();
for (int index=0; index< size; index++)
{
if (properties.elementAt(index) instanceof HTMLHyperlink)
{
link = (HTMLHyperlink)properties.elementAt(index);
break;
}
}
}
return link;
}
/**
* Returns the default HTML table.
* @return The table.
**/
public HTMLTable getTable()
{
return htmlTable_;
}
/**
* Initializes the transient data.
**/
//private void initializeTransient() @CRS
//{
//@CRS changes_ = new PropertyChangeSupport(this);
//@CRS vetos_ = new VetoableChangeSupport(this);
//@CRS sectionCompletedSupport_ = new SectionCompletedSupport(this);
//@CRS completedListeners_ = new Vector();
//}
/**
* Indicates whether the table header is created using the
* metadata. Default value is false (use existing table header).
* @return true if the metadata is used; false otherwise.
**/
public boolean isUseMetaData()
{
return useMetaData_;
}
/**
* Deserializes and initializes transient data.
**/
private void readObject(java.io.ObjectInputStream in)
throws java.io.IOException, ClassNotFoundException
{
in.defaultReadObject();
//initializeTransient(); @CRS
}
/**
* Removes this ActionCompletedListener from the internal list.
* If the ActionCompletedListener is not on the list, nothing is done.
* @param listener The ActionCompletedListener.
* @see #addActionCompletedListener
**/
public void removeActionCompletedListener(ActionCompletedListener listener)
{
if (listener == null)
throw new NullPointerException("listener");
if (completedListeners_ != null) completedListeners_.removeElement(listener); //@CRS
}
/**
* Removes the PropertyChangeListener from the internal list.
* If the PropertyChangeListener is not on the list, nothing is done.
* @param listener The PropertyChangeListener.
* @see #addPropertyChangeListener
**/
public void removePropertyChangeListener(PropertyChangeListener listener)
{
if (listener == null)
throw new NullPointerException("listener");
if (changes_ != null) changes_.removePropertyChangeListener(listener); //@CRS
}
/**
* Removes this SectionCompletedListener from the internal list.
* If the SectionCompletedListener is not on the list, nothing is done.
* @param listener The SectionCompletedListener.
* @see #addSectionCompletedListener
**/
public void removeSectionCompletedListener(SectionCompletedListener listener)
{
if(listener == null) //@KCA
throw new NullPointerException("listener"); //@KCA
if (sectionCompletedSupport_ != null) sectionCompletedSupport_.removeSectionCompletedListener(listener); //@CRS
}
/**
* Removes the VetoableChangeListener from the internal list.
* If the VetoableChangeListener is not on the list, nothing is done.
* @param listener The VetoableChangeListener.
* @see #addVetoableChangeListener
**/
public void removeVetoableChangeListener(VetoableChangeListener listener)
{
if (listener == null)
throw new NullPointerException("listener");
if (vetos_ != null) vetos_.removeVetoableChangeListener(listener); //@CRS
}
/**
* Sets the table header's hyperlinks.
* @param links The hyperlinks.
* @exception PropertyVetoException If a change is vetoed.
**/
public void setHeaderHyperlinks(HTMLHyperlink[] links) throws PropertyVetoException
{
if (links == null)
throw new NullPointerException("links");
HTMLHyperlink[] old = links_;
if (vetos_ != null) vetos_.fireVetoableChange("links", old, links); //@CRS
links_ = links;
if (changes_ != null) changes_.firePropertyChange("links", old, links); //@CRS
}
/**
* Sets the maximum number of rows in a table. The default value is 0 (no maximum).
* @param size The maximum size.
* @exception PropertyVetoException If a change is vetoed.
**/
public void setMaximumTableSize(int size) throws PropertyVetoException
{
if (size < 0)
throw new ExtendedIllegalArgumentException("size", ExtendedIllegalArgumentException.RANGE_NOT_VALID);
//@CRS Integer oldSize = new Integer(maxTableSize_);
//@CRS Integer newSize = new Integer(size);
int oldSize = maxTableSize_; //@CRS
if (vetos_ != null) vetos_.fireVetoableChange("size", new Integer(oldSize), new Integer(size)); //@CRS
maxTableSize_ = size;
if (changes_ != null) changes_.firePropertyChange("size", new Integer(oldSize), new Integer(size)); //@CRS
}
/**
* Sets the object's hyperlink at the specified column within the current row.
* The hyperlink is a property of the data object that can be used to link the data object
* to an Uniform Resource Identifier (URI).
* @param rowdata The RowData object that contains the rowdata.
* @param link The hyperlink tag.
* @param column The column number (0-based).
*
* @exception RowDataException If a row data exception occurs.
**/
public void setObjectHyperlink(RowData rowdata, HTMLHyperlink link, int column) throws RowDataException
{
// Validate the rowdata parameter.
if (rowdata == null)
throw new NullPointerException("rowdata");
setObjectHyperlink(rowdata, link, rowdata.getCurrentPosition(), column);
}
/**
* Sets the row object's hyperlink specified by row and column.
* The hyperlink is a property of the data object that can be used to link the data
* object to an Uniform Resource Identifier (URI).
* @param rowdata The RowData object that contains the row data.
* @param link The hyperlink tag.
* @param row The row number (0-based).
* @param column The column number (0-based).
* @exception RowDataException If a row data exception occurs.
**/
public void setObjectHyperlink(RowData rowdata, HTMLHyperlink link, int row, int column) throws RowDataException
{
// Validate the rowdata parameter.
if (rowdata == null)
throw new NullPointerException("rowdata");
// Validate the link parameter.
if (link == null)
throw new NullPointerException("link");
// Validate the row parameter.
if (!rowdata.absolute(row))
throw new ExtendedIllegalArgumentException("row", ExtendedIllegalArgumentException.RANGE_NOT_VALID);
// Get the object's properties.
Vector properties = rowdata.getObjectProperties(column);
// Add the hyperlink to the object properties list.
if (properties == null)
{
// Create the properties list and add link.
properties = new Vector();
properties.addElement(link);
}
else
{
// Has properties.
HTMLHyperlink old = null; // The existing hyperlink object.
int linkIndex = -1; // The property index of the the existing hyperlink.
// Check for existing hyperlink.
int size = properties.size();
for (int index=0; index< size; index++)
{
if (properties.elementAt(index) instanceof HTMLHyperlink)
{
// Get the existing hyperlink.
old = (HTMLHyperlink)properties.elementAt(index);
linkIndex = index;
break;
}
}
if (old == null)
properties.addElement(link);
else
properties.setElementAt(link, linkIndex);
}
// Set the row object's properties with the new hyperlink.
rowdata.setObjectProperties(properties, column);
}
/**
* Sets the default HTML table to be used during conversion.
* The default table's column headers must be set. The setUseMetaData
* method can also be used to set the column headers based on the metadata.
* @param table The HTML table.
* @exception PropertyVetoException If the change is vetoed.
* @see #setUseMetaData
**/
public void setTable(HTMLTable table) throws PropertyVetoException
{
if (table == null)
throw new NullPointerException("table");
HTMLTable old = htmlTable_;
if (vetos_ != null) vetos_.fireVetoableChange("table", old, table); //@CRS
htmlTable_ = table;
if (changes_ != null) changes_.firePropertyChange("table", old, table); //@CRS
}
/**
* Sets the table column header. The metadata column labels are
* used in creating the table header. If a column label does not
* exist the column name is used.
*
* @param metadata The meta data.
* @exception RowDataException If a row data error occurs.
* @exception PropertyVetoException If a property change is vetoed.
**/
private void setTableHeader(RowMetaData metadata)
throws RowDataException, PropertyVetoException
{
// Create the header list.
int numColumns = metadata.getColumnCount();
HTMLTableHeader[] headerList = new HTMLTableHeader[numColumns];
// Get the header names from the metadata.
String colName = "";
HTMLTagElement element;
for (int column=0; column< numColumns; column++)
{
// Use the column label if it exists; otherwise use the name.
try
{
colName = metadata.getColumnLabel(column);
}
catch (NullPointerException e)
{
colName = metadata.getColumnName(column);
}
// Check for hyperlinks.
if (links_ != null && links_[column] != null)
{
HTMLHyperlink link = links_[column];
link.setText(colName);
element = link;
}
else
element = new HTMLText(colName);
headerList[column] = new HTMLTableHeader(element);
}
// Set the table's header.
htmlTable_.setHeader(headerList);
if (!htmlTable_.isHeaderInUse())
htmlTable_.setHeaderInUse(true);
}
/**
* Sets whether the table header is created using the
* metadata. Default value is false (use existing table header).
* @param useMetaData true if the metadata is used; false otherwise.
**/
public void setUseMetaData(boolean useMetaData)
{
useMetaData_ = useMetaData;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy