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 Show documentation
Show all versions of jt400 Show documentation
The Open Source version of the IBM Toolbox for Java
///////////////////////////////////////////////////////////////////////////////
//
// 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