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

com.codename1.ui.html.ResourceThreadQueue Maven / Gradle / Ivy

/*
 * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores
 * CA 94065 USA or visit www.oracle.com if you need additional information or
 * have any questions.
 */
package com.codename1.ui.html;

import com.codename1.ui.Component;
import com.codename1.ui.Display;
import com.codename1.ui.Image;
import com.codename1.ui.Label;
import com.codename1.ui.geom.Dimension;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.Enumeration;
import java.util.Hashtable;
import java.util.Vector;

/**
 * ResourceThreadQueue is a thread queue used to create and manage threads that download images and CSS files that were referred from HTML pages
 * Was called ImageThreadQueue but name was changed since it now handles CSS as well
 *
 * @author Ofir Leitner
 */
class ResourceThreadQueue {

    /**
     * The default number of maximum threads used for image download
     */
    private static int DEFAULT_MAX_THREADS = 2;

    HTMLComponent htmlC;
    Vector queue = new Vector();
    Vector running = new Vector();
    Vector bgImageCompsUnselected = new Vector();
    Vector bgImageCompsSelected = new Vector();
    Vector bgImageCompsPressed = new Vector();

    Hashtable images = new Hashtable();
    static int maxThreads = DEFAULT_MAX_THREADS;
    int threadCount;
    private int cssCount=-1; // As long as there are no CSS files this stays -1 and thus CSS loading is ignored
    boolean started;

    /**
     * Constructs the queue
     * 
     * @param htmlC The HTMLComponent this queue belongs to
     */
    ResourceThreadQueue(HTMLComponent htmlC) {
        this.htmlC=htmlC;
    }

    /**
     * Sets the maximum number of threads to use for image download
     * If startRunning was already called, this will takes effect in the next page loaded.
     *
     * @param threadsNum the maximum number of threads to use for image download
     */
    static void setMaxThreads(int threadsNum) {
        maxThreads=threadsNum;
    }

    /**
     * Adds the image to the queue
     *
     * @param imgLabel The label in which the image should be contained after loaded
     * @param imageUrl The URL this image should be fetched from
     */
    synchronized void add(Component imgLabel,String imageUrl) {
        /*if ((HTMLComponent.TABLES_LOCK_SIZE) && (htmlC.curTable!=null)) {
            downloadImageImmediately(imgLabel, imageUrl, 0);
        } else {*/
        if (started) {
            throw new IllegalStateException("ResourceThreadQueue already started! stop/cancel first");
        }
        images.put(imgLabel, imageUrl); // Using a hashtable to collect all requests first enables overriding urls for labels (For example in CSS use cases)
        //}
    }

    /**
     * Adds the image to the queue, to be used as a background image for a component
     *
     * @param imgComp The component for which the image should be used after loaded
     * @param imageUrl The URL this image should be fetched from
     * @param styles A mask of CSSEngine.STYLE_* values indicating in which styles this background image should be displayed
     */
    synchronized void addBgImage(Component imgComp,String imageUrl,int styles) {
        if (HTMLComponent.SUPPORT_CSS) {
            /*if ((HTMLComponent.TABLES_LOCK_SIZE) && (htmlC.curTable!=null)) {
                downloadImageImmediately(imgComp,imageUrl,styles);
            } else {*/
                add(imgComp,imageUrl);
                if ((styles & CSSEngine.STYLE_SELECTED)!=0) {
                    bgImageCompsSelected.addElement(imgComp);
                }
                if ((styles & CSSEngine.STYLE_UNSELECTED)!=0) {
                    bgImageCompsUnselected.addElement(imgComp);
                }
                if ((styles & CSSEngine.STYLE_PRESSED)!=0) {
                    bgImageCompsPressed.addElement(imgComp);
                }
            //}
        }
    }

    /**
     * Downloads the image immediately // It seems that the issue that required this was already solved in the table package
     * The image is not added to the queue, and not loaded on another thread but rather downloaded on the same thread that builds up the document
     * This is useful for HTMLCoponent.TABLES_PATCH
     *
     * @param imgComp The component for which the image should be used after loaded
     * @param imageUrl The URL this image should be fetched from
     * @param styles A mask of CSSEngine.STYLE_* values indicating in which styles this background image should be displayed
     *
    synchronized void downloadImageImmediately(Component imgComp,String imageUrl,int styles) {
        try {
            InputStream is = htmlC.getRequestHandler().resourceRequested(new DocumentInfo(imageUrl,DocumentInfo.TYPE_IMAGE));
            Image img = Image.createImage(is);
            ResourceThread t = new ResourceThread(imageUrl, imgComp, htmlC, null);
            t.handleImage(img, imgComp,((styles & CSSEngine.STYLE_UNSELECTED)!=0),((styles & CSSEngine.STYLE_SELECTED)!=0),((styles & CSSEngine.STYLE_PRESSED)!=0));
        } catch (Exception ex) {
            ex.printStackTrace();
        }

    }*/

    /**
     * Adds a stylesheet to the queue
     *
     * @param cssUrl The URL this style sheet should be fetched from
     * @param dom the dom object on which the CSS should be applied
     */
    synchronized void addCSS(String cssUrl,String encoding) {
        if (started) {
            throw new IllegalStateException("ResourceThreadQueue alreadey started! stop/cancel first");
        }
        DocumentInfo cssDocInfo=new DocumentInfo(cssUrl, DocumentInfo.TYPE_CSS);
        if (encoding!=null) {
            cssDocInfo.setEncoding(encoding);
        }
        ResourceThread t =  new ResourceThread(cssDocInfo, htmlC, this);
        queue.addElement(t);
        incCSSCount();
    }

    /**
     * Incereases the internal count of the number of pending CSS documents
     */
    private void incCSSCount() {
        if (cssCount==-1) { // first CSS make sure we bump it from -1 to 1.
            cssCount++;
        }
        cssCount++;
    }

    /**
     * Returns the number of pending CSS documents
     * 
     * @return the number of pending CSS documents
     */
    synchronized int getCSSCount() {
        return cssCount;
    }

    /**
     * Returns the queue size, this is relevant only when the queue hasn't started yet
     *
     * @return the queue size
     */
    int getQueueSize() {
        return (images.size()+queue.size()); // CSS files are added directly to queue while images to the images vector
        //return queue.size();
    }

    /**
     * Notifies the queue that all images and CSS have been queues and it can start dequeuing and download the images.
     * The queue isn't started before that to prevent multiple downloads of the same image
     */
    synchronized void startRunning() {
        if (!startDequeue()) {
            startRunningImages();
        }
    }

    /**
     * Starts downloading the images (This is called only after all CSS files have been downloaded, since they may contain image references)
     */
    synchronized void startRunningImages() {
        queue.removeAllElements();
        Vector urls=new Vector();
        for(Enumeration e=images.keys();e.hasMoreElements();) {
            Component imgComp = (Component)e.nextElement();
            String imageUrl = (String)images.get(imgComp);
            int urlIndex=urls.indexOf(imageUrl);

            if (urlIndex!=-1) {
                ResourceThread t=(ResourceThread)queue.elementAt(urlIndex);
                t.addLabel(imgComp);
            } else {
                ResourceThread t =  new ResourceThread(imageUrl, imgComp, htmlC, this);
                queue.addElement(t);
                urls.addElement(imageUrl);
            }
        }
        urls=null;
        
        images=new Hashtable();

        if (!startDequeue()) {
            htmlC.setPageStatus(HTMLCallback.STATUS_COMPLETED);
        }

    }

    /**
     *  Starts dequeuing the queue into the running pool and launch them
     * 
     * @return true if there are at least one active thread, false otherwise
     */
    private synchronized boolean startDequeue() {
        int threads=Math.min(queue.size(), maxThreads);

        for(int i=0;i0);
    }

    /**
     * Called by the ResourceThread when it finishes downloading and setting the image.
     * This in turns starts another thread if the queue is not empty
     * 
     * @param finishedThread The calling thread
     * @param success true if the image download was successful, false otherwise
     */
    synchronized void threadFinished(ResourceThread finishedThread,boolean success) {
        
        if(finishedThread.cssDocInfo!=null) {
            cssCount--; // Reduce the number of waiting CSS, even if reading failed
        }

        if ((HTMLComponent.SUPPORT_CSS) && (cssCount==0)) {
            cssCount=-1; // So it won't get applied again
            htmlC.applyAllCSS();
            htmlC.cssCompleted();
        }
        running.removeElement(finishedThread);


        if (queue.size()>0) {
            ResourceThread t=(ResourceThread)queue.firstElement();
            queue.removeElementAt(0);
            running.addElement(t);
            t.go();          //new Thread(t).start();
        } else {
            threadCount--;
        }

        if (threadCount==0) {
            if (images.size()==0) {
                htmlC.setPageStatus(HTMLCallback.STATUS_COMPLETED);
            } else {
                startRunningImages();
            }
        }
    }

    /**
     * Discards the entire queue and signals the running threads to cancel.
     * THis will be triggered if the user cancelled the page or moved to another page.
     */
    synchronized void discardQueue() {
        queue.removeAllElements();

        for(Enumeration e=running.elements();e.hasMoreElements();) {
            ResourceThread t = (ResourceThread)e.nextElement();
            t.cancel();
        }
        running.removeAllElements();
        bgImageCompsSelected.removeAllElements();
        bgImageCompsUnselected.removeAllElements();
        bgImageCompsPressed.removeAllElements();

        threadCount=0;
        cssCount=-1;
        started=false;

    }

    /**
     * Returns a printout of the threads queue, can be used for debugging
     *
     * @return a printout of the threads queue
     */
    public String toString() {
        String str=("---- Running ----\n");
        int i=1;
        for(Enumeration e=running.elements();e.hasMoreElements();) {
            ResourceThread t = (ResourceThread)e.nextElement();
            if (t.imageUrl!=null) {
                str+="#"+i+": "+t.imageUrl+"\n";
            } else {
                str+="#"+i+": CSS - "+t.cssDocInfo.getUrl()+"\n";
            }
            i++;
        }
        i=1;
        str+="Queue:\n";
        for(Enumeration e=queue.elements();e.hasMoreElements();) {
            ResourceThread t = (ResourceThread)e.nextElement();
            if (t.imageUrl!=null) {
                str+="#"+i+": "+t.imageUrl+"\n";
            } else {
                str+="#"+i+": CSS - "+t.cssDocInfo.getUrl()+"\n";
            }
            i++;
        }
        str+="---- count:"+threadCount+" ----\n";
        return str;
    }

    // Inner classes:

    /**
     * An ResourceThread downloads an Image as requested
     *
     * @author Ofir Leitner
     */
    class ResourceThread implements Runnable, IOCallback {

        Component imgLabel;
        Vector labels;
        String imageUrl;
        DocumentRequestHandler handler;
        ResourceThreadQueue threadQueue;
        boolean cancelled;
        HTMLComponent htmlC;
        Image img;
        DocumentInfo cssDocInfo;

        /**
         * Constructs the ResourceThread for an image file
         *
         * @param imgLabel The label in which the image should be contained after loaded
         * @param imageUrl The URL this image should be fetched from
         * @param handler The RequestHandler through which to retrieve the image
         * @param threadQueue The main queue, for callback purposes
         */
        ResourceThread(String imageUrl, Component imgLabel,HTMLComponent htmlC,ResourceThreadQueue threadQueue) {
            this.imageUrl=imageUrl;
            this.imgLabel=imgLabel;
            this.handler=htmlC.getRequestHandler();
            this.threadQueue=threadQueue;
            this.htmlC=htmlC;
        }

        /**
         * Constructs the ResourceThread for a CSS file
         *
         * @param cssDocInfo A DocumentInfo object with the URL this CSS file should be fetched from
         * @param handler The RequestHandler through which to retrieve the image
         * @param threadQueue The main queue, for callback purposes
         */
        ResourceThread(DocumentInfo cssDocInfo,HTMLComponent htmlC,ResourceThreadQueue threadQueue) {
            this.cssDocInfo=cssDocInfo;
            this.handler=htmlC.getRequestHandler();
            this.threadQueue=threadQueue;
            this.htmlC=htmlC;
        }

        /**
         * Cancels this thread
         */
        void cancel() {
            cancelled=true;
        }

        /**
         * Adds a label which has the same URL, useful for duplicate images in the same page
         *
         * @param label A label which has the same image URL
         */
        void addLabel(Component label) {
            if (labels==null) {
                labels=new Vector();
            }
            labels.addElement(label);
        }

        /**
         * This is the main entry point to this runnable, it checks whether the callback is synchronous or async.
         * According to that it either runs this as a thread (sync) or simply calls the async method (async implements threading itself)
         */
        void go() {
                if (handler instanceof AsyncDocumentRequestHandler) {
                DocumentInfo docInfo=cssDocInfo!=null?cssDocInfo:new DocumentInfo(imageUrl,DocumentInfo.TYPE_IMAGE);
                    ((AsyncDocumentRequestHandler)handler).resourceRequestedAsync(docInfo, this);
                } else {
                    Display.getInstance().startThread(this, "HTML Resources").start();
                }
        }

        /**
         * {{@inheritDoc}}
         */
        public void run() {
                DocumentInfo docInfo=cssDocInfo!=null?cssDocInfo:new DocumentInfo(imageUrl,DocumentInfo.TYPE_IMAGE);
                InputStream is = handler.resourceRequested(docInfo);
                streamReady(is, docInfo);
        }

        /**
         * {{@inheritDoc}}
         */
        public void streamReady(InputStream is,DocumentInfo docInfo) {
            try {
                if (is==null) {
                    if (htmlC.getHTMLCallback()!=null) {
                        htmlC.getHTMLCallback().parsingError(cssDocInfo!=null?HTMLCallback.ERROR_CSS_NOT_FOUND:HTMLCallback.ERROR_IMAGE_NOT_FOUND, null, null, null, (cssDocInfo!=null?"CSS":"Image")+" not found at "+(cssDocInfo!=null?cssDocInfo.getUrl():imageUrl));
                    }
                } else {
                    if(cssDocInfo!=null) { // CSS
                        if (HTMLComponent.SUPPORT_CSS) { // no need to also check if loadCSS is true, since if we got so far - it is...
                            CSSElement result = CSSParser.getInstance().parseCSSSegment(new InputStreamReader(is),is,htmlC,cssDocInfo.getUrl());
                            result.setAttribute(result.getAttributeName(new Integer(CSSElement.CSS_PAGEURL)), cssDocInfo.getUrl());
                            htmlC.addToExternalCSS(result);
                        }
                        threadQueue.threadFinished(this,true);
                        return;
                    } else {
                        img=Image.createImage(is);
                        if (img==null) {
                            if (htmlC.getHTMLCallback()!=null) {
                                htmlC.getHTMLCallback().parsingError(HTMLCallback.ERROR_IMAGE_BAD_FORMAT, null, null, null, "Image could not be created from "+imageUrl);
                            }
                        }
                    }
                }

                if (img==null) {
                    threadQueue.threadFinished(this,false);
                    return;
                }
                if (!cancelled) {
                    Display.getInstance().callSerially(new Runnable() {
                        public void run() {
                            // prevent a redirect or another thread from breaking the UI
                            handleImage(img,imgLabel);
                            if (labels!=null) {
                                for(Enumeration e=labels.elements();e.hasMoreElements();) {
                                    Component cmp=(Component)e.nextElement();
                                    handleImage(img,cmp);
                                }
                            }
                        }
                    });
                    threadQueue.threadFinished(this,true);
                }
            } catch (IOException ioe) {
                if (htmlC.getHTMLCallback()!=null) {
                    htmlC.getHTMLCallback().parsingError(HTMLCallback.ERROR_IMAGE_BAD_FORMAT, null, null, null, "Image could not be created from "+imageUrl+": "+ioe.getMessage());
                }
                if(!cancelled) {
                    threadQueue.threadFinished(this,false);
                }
            }

        }

        /**
         * After a successful download, this handles placing the image on the label and resizing if necessary
         *
         * @param img The image
         * @param label The label to apply the image on
         */
        private void handleImage(Image img,Component cmp) {
            boolean bgUnselected=(threadQueue.bgImageCompsUnselected.contains(cmp));;
            boolean bgSelected=(threadQueue.bgImageCompsSelected.contains(cmp));
            boolean bgPressed=(threadQueue.bgImageCompsPressed.contains(cmp));
            handleImage(img, cmp,bgUnselected,bgSelected,bgPressed);
        }

        /**
         * After a successful download, this handles placing the image on the label and resizing if necessary
         *
         * @param img The image
         * @param cmp The component to apply the image on
         * @param bgUnselected true if the image should be used as a background for the component when it is unselected, false otherwise
         * @param bgSelected true if the image should be used as a background for the component when it is selected, false otherwise
         * @param bgPressed true if the image should be used as a background for the component when it is pressed, false otherwise
         */
        void handleImage(Image img,Component cmp,boolean bgUnselected,boolean bgSelected,boolean bgPressed) {
            boolean bg=false;
            if (bgUnselected) {
                cmp.getUnselectedStyle().setBgImage(img);
                bg=true;
            }
            if (bgSelected) {
                cmp.getSelectedStyle().setBgImage(img);
                bg=true;
            }
            if (bgPressed) {
                if (cmp instanceof HTMLLink) {
                    ((HTMLLink)cmp).getPressedStyle().setBgImage(img);
                }
                bg=true;
            }
            if (bg) {
                cmp.repaint();
                return;
            }

            Label label = (Label)cmp;
            label.setText(""); // remove the alternate text (important to do here before checking the width/height)

            // Was set in HTMLComponent.handleImage if the width/height attributes were in the tag
            int width=label.getPreferredW()-label.getStyle().getPadding(Component.LEFT)-label.getStyle().getPadding(Component.RIGHT);
            int height=label.getPreferredH()-label.getStyle().getPadding(Component.TOP)-label.getStyle().getPadding(Component.BOTTOM);

            if (width!=0) {
                if (height==0) { // If only width was specified, height should be calculated so the image keeps its aspect ratio
                    height=img.getHeight()*width/img.getWidth();
                }
            } else if (height!=0) {
                if (width==0) { // If only height was specified, width should be calculated so the image keeps its aspect ratio
                    width=img.getWidth()*height/img.getHeight();
                }
            }

            if (width!=0) { // if any of width or height were not 0, the other one was set to non-zero above, so this check suffices
                img=img.scaled(width, height);
                width+=label.getStyle().getPadding(Component.LEFT)+label.getStyle().getPadding(Component.RIGHT);
                height+=label.getStyle().getPadding(Component.TOP)+label.getStyle().getPadding(Component.BOTTOM);
                label.setPreferredSize(new Dimension(width,height));
            }

            label.setIcon(img);

            htmlC.revalidate();
            if (label.getClientProperty(HTMLComponent.CLIENT_PROPERTY_IMG_BORDER)==null) { // if this property is defined, it means the image had a border already
                label.getUnselectedStyle().setBorder(null); //remove the border which is a sign the image is loading
            } else {
                int borderSize=((Integer)label.getClientProperty(HTMLComponent.CLIENT_PROPERTY_IMG_BORDER)).intValue();
                // Note that padding is set here and not in handleImage since we rely on the image size (which includes padding) to know if a width/height was specified
                label.getUnselectedStyle().setPadding(borderSize,borderSize,borderSize,borderSize);
                label.getSelectedStyle().setPadding(borderSize,borderSize,borderSize,borderSize);
            }

        }

    }
    
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy