All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jclarion.clarion.print.OpenReport Maven / Gradle / Ivy

/**
 * Copyright 2010, by Andrew Barnham
 *
 * The contents of this file are subject to
 * GNU Lesser General Public License (LGPL), v.3
 * http://www.gnu.org/licenses/lgpl.txt
 * 
 * Software distributed under the License is distributed on an "AS IS" basis,
 * WITHOUT WARRANTY OF ANY KIND, either express or implied.
 */
package org.jclarion.clarion.print;

import java.awt.Font;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Point;
import java.awt.print.PageFormat;
import java.awt.print.Paper;
import java.awt.print.PrinterException;
import java.awt.print.PrinterJob;
import java.util.ArrayList;
import java.util.IdentityHashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import javax.print.attribute.PrintRequestAttributeSet;
import javax.print.attribute.standard.MediaSize;
import javax.print.attribute.standard.MediaSizeName;
import javax.print.attribute.standard.OrientationRequested;

import org.jclarion.clarion.Clarion;
import org.jclarion.clarion.ClarionObject;
import org.jclarion.clarion.ClarionPrinter;
import org.jclarion.clarion.ClarionReport;
import org.jclarion.clarion.PropertyObject;
import org.jclarion.clarion.PropertyObjectListener;
import org.jclarion.clarion.constants.Icon;
import org.jclarion.clarion.constants.Prop;
import org.jclarion.clarion.constants.Propprint;
import org.jclarion.clarion.control.AbstractControl;
import org.jclarion.clarion.control.AbstractReportControl;
import org.jclarion.clarion.control.ControlIterator;
import org.jclarion.clarion.control.ReportBreak;
import org.jclarion.clarion.control.ReportContainer;
import org.jclarion.clarion.control.ReportDetail;
import org.jclarion.clarion.control.ReportFooter;
import org.jclarion.clarion.control.ReportForm;
import org.jclarion.clarion.control.ReportHeader;
import org.jclarion.clarion.control.StringControl;
import org.jclarion.clarion.runtime.CMemory;
import org.jclarion.clarion.runtime.CWin;

/**
 * Manage printing of report
 * 
 * Some notes about clarion reports:
 *   Withprior and withnext are cumulative.
 *   
 *   withprior and withnext will not force a page to completely empty
 *
 *   In clarion implementation - withprior will consider add added
 *   item and recursively scan back. If recursive scan back results in
 *   prior page being completely emptied, then withprior is ignored
 *   
 *   In clarion implementation - withnext appears to trigger when
 *   the page is closed. System will identify items on page to move onto
 *   next page based on withnext setting. If withnext would result in 
 *   page being completely emptied then withnext skips
 * 
 *   Group headers and footers are printed on use var breaks only and
 *   does not pay attention to page breaks
 *
 *   For aggregate activity as follows:
 *   ---
 *   Firstly the rules taken from clarion documentation:
 *    Options include: SUM, COUNT, MAX, MIN, AVE
 *    
 *    When in a detail structure - is evaluated every time detail structure prints
 *    so provides running value
 *    
 *    In group footer - based on detail structures for any detail within the break
 *    that is printed.
 *    
 *    In footer - based on any detail on that page only
 *    
 *   RESET option specifies break control on which statistic resets.
 *   
 *   TALLY option specifies when counting occurs.
 *   
 *   PAGE option specifies that statistic resets when page break occurs.
 *   
 *   PAGENO option specifies that string is to contain the page number
 *
 * There are a couple of aspects of the implementation that I am not fully happy
 * with. Main item is how paging is setup. Paging occurs after individual printing
 * blocks - details, headers, footers, etc are laid out. As such a second cycle
 * occurs where things like PROP:PAGENO are recalculated and also SUM, CNT etc are
 * implemented on page footers. Result is that following are misimplemented:
 *   PROP:PAGE does not work at all
 *   RESET/TALLY on page footers do not work at all 
 *   
 * Ideally we want to do two passes - pass one does basic layouts taking items
 * like withprior and withnext into consideration and  and pass two performs 
 * aggregation and aggregation can also tally/reset based on page break activity result
 * is that PAGE becomes a meaningful reset property for all aggregations and
 * all agregations are implemented the same - unlike right now where aggregations in
 * detail and break footers are different from aggregations in the page footer.
 * 
 * But there are even challenges with this - because aggregate result and influence
 * how blocks on a page layout and thus influence page break triggers. i.e. you can
 * setup a break on VARX and VARX forms MAX(VARX) so aggregate triggers decision
 * to break or not which in turn influences page layout in this case - printing of 
 * break headers/footers. Problem - if required to be solved - needs more thought.
 * 
 * @author barney
 *
 */

public class OpenReport implements PropertyObjectListener
{
    public static ReportDetail NEWPAGE = new ReportDetail();

    private static class Element
    {
        private PrintObject detail;
        private Element prior;
        private Element next;
    }

    
    private ClarionReport report;
    
    private int x;
    private int y;
    private int width;
    private int height;
    
    private double xscale;  // adjustors to convert from clarion size to /72th of an inch
    private double yscale;  // adjustors to convert from clarion size to /72th of an inch
    
    private PrinterJob job;
    private PrintRequestAttributeSet printAttribute;

    private Map tally=new IdentityHashMap(); 
    private Map reset=new IdentityHashMap(); 

    private Map stats=new IdentityHashMap();

    private Map isAggregatingField=new IdentityHashMap();

    private Map breakValues=new IdentityHashMap();
    
    private Element firstDetail;
    private Element lastDetail;
    private Element position;

    private List pages;
    
    private PrintContext graphics;
    private PageFormat   page;

    private ReportDetail lastPrintedItem;

    private Page currentPage;
    
    
    public OpenReport(ClarionReport report)
    {
        this.report=report;
        
        xscale=1;
        yscale=1;
        
        if (report!=null) {
            x=report.getProperty(Prop.XPOS).intValue();
            y=report.getProperty(Prop.YPOS).intValue();
            width=report.getProperty(Prop.WIDTH).intValue();
            height=report.getProperty(Prop.HEIGHT).intValue();

            if (report.isProperty(Prop.THOUS)) {
                xscale=yscale=72.0/1000;
            } else if (report.isProperty(Prop.MM)) {
                xscale=yscale=72.0/25.4;
            } else if (report.isProperty(Prop.POINTS)) {
            
            } else {
                Font f = CWin.getInstance().getFontOnly(null,report);
                if (f!=null) {
                    xscale=f.getSize()/4.0;
                    yscale=f.getSize()/8.0;
                }
            }
            
            job = ClarionPrinter.getInstance().getJob();
            if (job!=null) {
                job.setJobName(report.getProperty(Prop.TEXT).toString());
                determineAttributes();
            }
            
            if (report.getPreview()!=null) {
                init(ClarionPrinter.getInstance().getGraphics());
            }
        }
        
        xscale=xscale*10.0;
        yscale=yscale*10.0;
        
        refreshTallyDependencies();
    }

    public ClarionReport getReport()
    {
        return report;
    }
    
    @Override
    public Object getProperty(PropertyObject owner, int property) {
        return null;
    }

    @Override
    public void propertyChanged(PropertyObject owner, int property,
            ClarionObject value) {
        switch(property) {
            case Prop.LANDSCAPE:
                determineAttributes();
                break;
            case Prop.XPOS:
                x=value.intValue();
                break;
            case Prop.YPOS:
                y=value.intValue();
                break;
            case Prop.WIDTH:
                width=value.intValue();
                break;
            case Prop.HEIGHT:
                height=value.intValue();
                break;
            case Prop.TEXT:
                if (job!=null) {
                    job.setJobName(value.toString().trim());
                }
        }
    }
    
    
    private void determineAttributes() {
        // work out paper
        printAttribute = ClarionPrinter.getInstance().getAttribute();
        if (report.isProperty(Prop.LANDSCAPE)) {
            printAttribute.add(OrientationRequested.LANDSCAPE);
        } else {
            printAttribute.add(OrientationRequested.PORTRAIT);
        }
    
        int type = report.getProperty(Propprint.PAPER).intValue();
        switch(type) {
            case org.jclarion.clarion.constants.Paper.A4:
                printAttribute.add(MediaSizeName.ISO_A4);
                break;
            case org.jclarion.clarion.constants.Paper.LEGAL:
                printAttribute.add(MediaSizeName.NA_LEGAL);
                break;
            case org.jclarion.clarion.constants.Paper.LETTER:
                printAttribute.add(MediaSizeName.NA_LETTER);
                break;
            case org.jclarion.clarion.constants.Paper.USER:
            {
                int pw=report.getProperty(Propprint.PAPERWIDTH).intValue();
                int ph=report.getProperty(Propprint.PAPERHEIGHT).intValue();
                
                try {
                    printAttribute.add(new MediaSize(
                        (float)(pw*72.0*xscale),
                        (float)(ph*72.0*yscale),
                        MediaSize.INCH));
                } catch (Exception ex) { }
                break;
            }
            case 0:
                break;
            default:
                throw new IllegalStateException("Paper Type not supported");
        }
        page = job.getPageFormat(printAttribute);
    }

    
    public void registerTallyListener(AbstractReportControl control,ReportStatistic statistic)
    {
        register(tally,control,statistic);
    }

    public void registerResetListener(AbstractReportControl control,ReportStatistic statistic)
    {
        register(reset,control,statistic);
    }
    
    private void register(
            Map type,
            AbstractReportControl control, ReportStatistic statistic) 
    {
        
        ReportStatistics rs = type.get(control);
        if (rs==null) {
            rs=new ReportStatistics();
            type.put(control,rs);
        }
        
        rs.add(statistic);
    }

    
    private AbstractReportControl getReportControl(AbstractControl control)
    {
        while ( control!=null ) {
            if (control instanceof AbstractReportControl) return (AbstractReportControl)control;
            control=control.getParent();
        }
        return null;
    }

    
    public boolean isAggregatingField(ClarionObject field)
    {
        return isAggregatingField.containsKey(field);
    }
    
    public ReportStatistic getStatistic(StringControl control)
    {
        ReportStatistic rs = stats.get(control);
        if (rs!=null) return rs;

        if (control.isProperty(Prop.SUM)) {
            rs=ReportStatistic.SUM(null);
        }

        if (control.isProperty(Prop.AVE)) {
            rs=ReportStatistic.AVG(null);
        }

        if (control.isProperty(Prop.CNT)) {
            rs=ReportStatistic.COUNT(null);
        }

        if (control.isProperty(Prop.MIN)) {
            rs=ReportStatistic.MIN(null);
        }

        if (control.isProperty(Prop.MAX)) {
            rs=ReportStatistic.MAX(null);
        }

        if (rs==null) return null;
        stats.put(control, rs);
        rs.setControl(control);
        isAggregatingField.put(control.getUseObject(),true);

        AbstractReportControl container = getReportControl(control);

        // if page footer
        if ((container instanceof ReportFooter)
            && !(container.getContainer() instanceof ReportBreak)) {
            
            rs.setPagedItem();
            
            return rs;
        }
        
        
        // work out tally
        AbstractReportControl tally = control.getTally();
        if (tally != null) {
            registerTallyListener(tally, rs);
        } else {
            
            if (container instanceof ReportDetail) {
                // When in a detail structure - is evaluated every time detail
                // structure prints
                
                registerTallyListener(container, rs);

            } else if ((container instanceof ReportFooter)
                    && (container.getContainer() instanceof ReportBreak)) {

                // In group footer - based on detail structures for any detail
                // within the break
                // that is printed.

                ReportComponentIterator rci = new ReportComponentIterator(
                        (ReportBreak) container.getContainer());
                while (rci.hasNext()) {
                    AbstractReportControl arc = rci.next();
                    if (arc instanceof ReportDetail) {
                        if (arc.containsUse(control.getUseObject())) {
                            registerTallyListener((ReportDetail) arc, rs);
                        }
                    }
                }
            }
        }

        // work out reset
        AbstractReportControl reset = control.getReset();
        if (reset != null) {
            registerResetListener(reset, rs);
        } else {

            if (container instanceof ReportDetail) {
                // never resets
            } else if ((container instanceof ReportFooter)
                    && (container.getContainer() instanceof ReportBreak)) {
                // In group footer - when break resets
                registerResetListener((ReportBreak) container.getContainer(),
                        rs);
            }
        }

        return rs;
    }
    
    
    public void pageBreak()
    {
        pageBreak(false);
    }

    public void pageBreak(boolean open)
    {
        if (!open) {
            printBreaks(null);
        }
        print(NEWPAGE);
    }
    

    public ReportContainer commonAncestor(ReportDetail d1,ReportDetail d2)
    {
        if (d1==null || d2==null) return report;
        
        Map s = new IdentityHashMap();
        
        ReportContainer s1 = d1.getContainer();
        ReportContainer s2 = d1.getContainer();
        
        while (s1 != null && s2 != null ) {
            
            if (s1==s2) return s1;
            
            if (s1!=null && s.containsKey(s1)) return s1;
            if (s2!=null && s.containsKey(s2)) return s2;
            
            if (s1!=null) {
                s.put(s1,null);
                s1=s1.getContainer();
            }
            if (s2!=null) {
                s.put(s2,null);
                s2=s2.getContainer();
            }
        }
        
        return report;
    }
    
    
    private void printBreaks(ReportDetail detail)
    {
        ReportContainer commonAncestor = commonAncestor(detail,lastPrintedItem);
        
        // footers - if break value has changed
        if (lastPrintedItem!=null) {
            ReportContainer scan = lastPrintedItem.getContainer();
            
            // force closure until common ancestor is reached
            while (scan!=commonAncestor && scan!=null && (scan instanceof ReportBreak)) 
            {
                ReportBreak b_scan = (ReportBreak)scan;
                add(b_scan.getFooter());
                
                ReportStatistics rs;
                rs = reset.get(b_scan);
                if (rs!=null) rs.reset();
                rs = tally.get(b_scan);
                if (rs!=null) rs.add();
                
                breakValues.remove(b_scan);
                scan=scan.getContainer();
            }
            
            // build up list of break
            ArrayList al = new ArrayList();
            while (scan!=null && (scan instanceof ReportBreak)) {
                al.add( (ReportBreak)scan );
                scan=scan.getContainer();
            }
            
            // work out which breaks need resetting
            if (!al.isEmpty()) {
                boolean resetAll=false;
                
                boolean reset[] = new boolean[al.size()];
                
                for (int r_scan = reset.length-1;r_scan>=0;r_scan--) {
                    ReportBreak rb = al.get(r_scan);
                    if (!resetAll) {
                        ClarionObject lastValue = breakValues.get(rb);
                        if (lastValue!=null) {
                            if (!lastValue.equals(rb.getUseObject())) {
                                resetAll=true;
                            }
                        }
                    }
                    if (resetAll) {
                        reset[r_scan]=true;
                    }
                }
                
                for (int r_scan=0;r_scan al = new ArrayList();
            ReportContainer scan = detail.getContainer();
            while (scan!=null) {
                if (scan instanceof ReportBreak) {
                    al.add((ReportBreak)scan);
                }
                scan=scan.getContainer();
            }

            for (int bscan=al.size()-1;bscan>=0;bscan--) {
                ReportBreak rb = al.get(bscan);
                if (breakValues.get(rb)==null) {
                    breakValues.put(rb,rb.getUseObject().genericLike());
                    
                    add(rb.getHeader());
                }
            }
        }
        
        // finally 
        lastPrintedItem = detail;
    }
    
    public void print(ReportDetail detail)
    {
        // handle group headers and footers
        if (detail!=NEWPAGE) {
            printBreaks(detail);
        }
        
        add(detail);
        printItems();
    }
    
    private void add(AbstractReportControl detail)
    {
        if (detail==null) return;
        
        ReportStatistics rs;
        
        rs = tally.get(detail);
        if (rs!=null) {
            rs.add();
        }

        rs = reset.get(detail);
        if (rs!=null) rs.reset();
        
        Element e = new Element();
        e.detail=new PrintObject(detail,this,CWin.getInstance());
        e.prior=lastDetail;
        e.next=null;
        
        if (lastDetail==null) {
            lastDetail=e;
            firstDetail=e;
        } else {
            lastDetail.next=e;
        }
        
        lastDetail=e;
    }
    
    public void reset()
    {
        position=null;
    }
    
    public PrintObject next()
    {
        if (position==null) {
            position=firstDetail;
        } else {
            position=position.next;
        }
        if (position==null) return null;
        return position.detail;
    }

    public PrintObject previous()
    {
        if (position==null) {
            position=lastDetail;
        } else {
            position=position.prior;
        }
        if (position==null) return null;
        return position.detail;
    }
    
    public PrintObject current()
    {
        if (position==null) return null;
        return position.detail;
    }
    
    public PrintObject peek(int offset)
    {
        Element scan = position;
        if (scan==null) {
            if (offset>0) {
                scan=firstDetail;
                offset--;
            } else {
                scan=lastDetail;
                offset++;
            }
        }
        
        while (offset>0 && scan!=null) {
            scan=scan.next;
            offset--;
        }
        while (offset<0 && scan!=null) {
            scan=scan.prior;
            offset++;
        }
        if (scan==null) return null;
        return scan.detail;
    }

    public Iterable getPages()
    {
        if (pages==null) init();
        return pages;
    }
   
    /*
    public Page getPage(int page)
    {
        if (pages==null) init();
        return pages.get(page);
    }
    */
    
    /**
    public int getPageCount()
    {
        if (pages==null) init();
        return pages.size();
    }*/

    public void refreshTallyDependencies()
    {
        if (report==null) return;
        ControlIterator ci = new ControlIterator(report);
        ci.setLoop(false);
        ci.setScanDisabled(true);
        ci.setScanHidden(true);
        ci.setScanSheets(true);
        while (ci.hasNext()) {
            AbstractControl ac = ci.next();
            if (ac instanceof StringControl) {
                getStatistic((StringControl)ac);
            }
        }
    }


    public PageFormat getPageFormat()
    {
        if (page==null) {
            PageFormat pf = new PageFormat();
            
            float f[] = MediaSize.ISO.A4.getSize(MediaSize.INCH);
            
            Paper paper = new Paper();
            paper.setSize(f[0]*72,f[1]*72);
            paper.setImageableArea(0,0,f[0]*72,f[1]*72);
            pf.setPaper(paper);
            if (report!=null) {
                pf.setOrientation(report.isProperty(Prop.LANDSCAPE) ? PageFormat.LANDSCAPE : PageFormat.PORTRAIT );
            }
            return pf;
        }
        return page;
    }
    
    public boolean isInit()
    {
        return (pages!=null);
    }
    
    public void init()
    {
        init(ClarionPrinter.getInstance().getGraphics());
    }
    
    public void init(Graphics graphics)
    {
        if (pages!=null) return;
        init(new AWTPrintContext((Graphics2D)graphics));
    }
        
    public void init(PrintContext graphics)
    {
        this.graphics=graphics;
        if (pages!=null) return;
        pages=new ArrayList();
        currentPage = null;
        reset();
        printItems();
    }

    private void finishPrint()
    {
        printBreaks(null);
        printItems();
        currentPage=finalizePage(currentPage);
        currentPage=finalizePage(currentPage);
    }
    

    private void printItems()
    {
        if (pages==null) return; // not yet!
        
        while (peek(1)!=null) {
            next();
            
            if (current().getControl()==NEWPAGE) {
                currentPage=finalizePage(currentPage,true);
                continue;
            }
            currentPage = doAddElement(currentPage,current());
        }
    }
    
    private Page doAddElement(Page currentPage, PrintObject current) 
    {
        if (currentPage==null) {
            currentPage=doNewPage();  
        }
        
        if (current.getControl().isProperty(Prop.PAGEBEFORE)) {
        	int count = current.getControl().getProperty(Prop.PAGEBEFORENUM).intValue();
        	if (count!=0) {
        		if (currentPage.getMoveableCount()>0) {
                    currentPage = finalizePage(currentPage);
                    if (currentPage==null) {
                    	currentPage=doNewPage();
                    }
        			currentPage.setReNumber(count);
        		}
        	}
        }
        
        if (!currentPage.layout(graphics,current)) {
            currentPage = finalizePage(currentPage);
            if (currentPage==null) {
                currentPage = doNewPage();
            }
            currentPage.layout(graphics,current);
        }
        
        ClarionObject o_prior = current.getControl().getRawProperty(Prop.WITHPRIOR);
        if (o_prior!=null && o_prior.intValue()>0) {
            int prior=o_prior.intValue();
            
            if (prior>=currentPage.getMoveableCount() && pages.size()>0) {
                int raid = prior-currentPage.getMoveableCount()+1;
                
                Page priorPage = pages.get(pages.size()-1);
                PrintObject priorBits[] = priorPage.getMoveableObjects();
                
                int scan=priorBits.length-1;
                
                // end represents the index of first element to put on
                // the next page
                int end = priorBits.length-raid;
                
                // scan through items checking for cumulative priors
                while (scan>=0) {
                    if (scan0) {

                    // create a new page and have a go add adding 
                    // prior bits and new bits. If not all fit then
                    // we want to rollback
                    
                    boolean rollback=false;
                    Page altPage = doNewPage();

                    PrintObject currentBits[] = currentPage.getMoveableObjects();
                    ArrayList points = new ArrayList(priorBits.length-end+currentBits.length);
                    
                    for (scan=end;scan rscan=points.iterator();
                        for (scan=end;scan0) {
                    currentPage = finalizePage(currentPage);
                    if (currentPage==null) {
                    	currentPage=doNewPage();
                    }
        			currentPage.setReNumber(count);
        		}
        	}
        }
        
        return currentPage;
    }

    private Page finalizePage(Page currentPage)
    {
        return finalizePage(currentPage,false);
    }
    
    private Page finalizePage(Page currentPage,boolean force)
    {
        if (currentPage==null) return null;
        ReportFooter rf = report.getFooter();
        if (rf!=null) {
            currentPage.add(graphics,new PrintObject(rf,this,CWin.getInstance(),false));
        }
        if (force || currentPage.getMoveableCount()>0) {
            
            if (currentPage.getReNumber()==-1) {
            	int lastPage=0;
            	if (!pages.isEmpty()) {
            		lastPage=pages.get(pages.size()-1).getPageNo();
            	}
            	currentPage.setPageNo(lastPage+1);
            } else {
            	currentPage.setPageNo(currentPage.getReNumber());
            }
            pages.add(currentPage);

            if (report.getPreview()!=null) {
                report.getPreview().what(1).setValue("print:/"+CMemory.address(currentPage));
                report.getPreview().setNextAnchor(currentPage);
                report.getPreview().add();
            }
            
            // handle withnext 
            PrintObject bits[] = currentPage.getMoveableObjects();
            
            int end = bits.length;
            
            for (int scan=bits.length-1;scan>=0;scan--) {
                int withNext = bits[scan].getControl().getProperty(Prop.WITHNEXT).intValue();
                if (withNext+scan0 && end




© 2015 - 2025 Weber Informatics LLC | Privacy Policy