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

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

There is a newer version: 7.0.167
Show newest version
/*
 * 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.Container;
import com.codename1.ui.Graphics;
import com.codename1.ui.Image;
import com.codename1.ui.Painter;
import com.codename1.ui.geom.Rectangle;
import com.codename1.ui.plaf.Style;

/**
 * A painter that draws the background of a component based on its CSS style
 * Unlike the regular CodenameOne bg painter, this also adds the following:
 * - Painting both a background color and a background image
 * - Tiling horizontally or vertically (or both)
 * - Background image position offset horizontal and vertical
 *
 * @author Ofir Leitner
 */
class CSSBgPainter implements Painter {
    private Component parent,scrollableParent;
    int horizPos,vertPos;
    boolean horizIsPercentage,vertIsPercentage;
    boolean fixedX;

    /**
     * Construct a background painter for the given component
     *
     * @param the parent component
     */
    CSSBgPainter(Component parent) {
        this.parent = parent;
    }

    /**
     * Sets the horizontal/vertical position of the background image
     *
     * @param property CSSElement.CSS_BACKGROUND_POSITION_X or Y for horizontal/vertical
     * @param pos The position as a CSS length value
     */
    void setPosition(int property,int pos) {
        boolean isPercentage=false;
        if ((pos & CSSElement.VAL_PERCENTAGE)!=0) {
            pos-=CSSElement.VAL_PERCENTAGE;
            isPercentage=true;
        }
        if (property==CSSElement.CSS_BACKGROUND_POSITION_X) {
            setHorizPosition(pos, isPercentage);
        } else {
            setVertPosition(pos, isPercentage);
        }
    }

    /**
     * Sets the horizontal position of the background image
     * 
     * @param pos The position either in pixels or percentage
     * @param isPercentage if true then the numeric value give at 'pos' is actually percentages
     */
    private void setHorizPosition(int pos,boolean isPercentage) {
        horizPos=pos;
        horizIsPercentage=isPercentage;
    }

    /**
     * Sets the vertical position of the background image
     *
     * @param pos The position either in pixels or percentage
     * @param isPercentage if true then the numeric value give at 'pos' is actually percentages
     */
    private void setVertPosition(int pos,boolean isPercentage) {
        vertPos=pos;
        vertIsPercentage=isPercentage;
    }

    /**
     * Sets this background to be fixed, which means it isn't affected by scrolling, but stays fixed in the back of the component it is applied on.
     */
    void setFixed() {
        fixedX=true;
    }

    /**
     * Returns a parent container that is scrollable or null if no parent is
     * scrollable.
     * This was copied from Container.getScrollableParent (Which can't be used as it is private)
     *
     * @return a parent container that is scrollable or null if no parent is
     * scrollable.
     */
    private Container getScrollableParent(Component parent) {
        Container cont;
        if (parent instanceof Container) {
            cont=(Container)parent;
        } else {
            cont=parent.getParent();
        }
        while (cont != null) {
            if ((cont.isScrollableX()) || (cont.isScrollableY())) { 
                return cont;
            }
            cont = cont.getParent();
        }
        return null;
    }

    /**
     * {@inheritDoc}
     */
    public void paint(Graphics g, Rectangle rect) {
        Style s = parent.getStyle();
        int x = rect.getX();
        int y = rect.getY();
        int width = rect.getSize().getWidth();
        int height = rect.getSize().getHeight();
        if (width <= 0 || height <= 0) {
            return;
        }
        g.setColor(s.getBgColor());
        g.fillRect(x, y, width, height, s.getBgTransparency());

        Image bgImage = s.getBgImage();

        if (bgImage == null) {
            return;
        }


        if (fixedX) {
            if (scrollableParent==null) {
                scrollableParent=getScrollableParent(parent);
            }
            if (scrollableParent!=null) {
                x+=scrollableParent.getScrollX();
                y+=scrollableParent.getScrollY();
                width=scrollableParent.getWidth();
                height=scrollableParent.getHeight();
            }
        }


        int iW=bgImage.getWidth();
        int iH=bgImage.getHeight();

        int offsetX=horizPos;
        int offsetY=vertPos;
        if (horizIsPercentage) {
            offsetX=(width-iW)*offsetX/100;
        }
        if (vertIsPercentage) {
            offsetY=(height-iH)*offsetY/100;
        }


        switch(s.getBackgroundType()) {
            case 0:
                g.drawImage(s.getBgImage(), x+offsetX, y+offsetY);
                return;
            case Style.BACKGROUND_IMAGE_TILE_BOTH:
                for (int xPos = getTiledPosition(offsetX, iW); xPos <= width; xPos += iW) {
                    for (int yPos = getTiledPosition(offsetY, iH); yPos <= height; yPos += iH) {
                        g.drawImage(s.getBgImage(), x + xPos, y + yPos);
                    }
                }
                return;
            case Style.BACKGROUND_IMAGE_TILE_HORIZONTAL:
                for (int xPos=getTiledPosition(offsetX, iW); xPos <= width; xPos += iW) {
                    g.drawImage(s.getBgImage(), x + xPos, y+offsetY);
                }
                return;
            case Style.BACKGROUND_IMAGE_TILE_VERTICAL:
                for (int yPos = getTiledPosition(offsetY, iH); yPos <= height; yPos += iH) {
                    g.drawImage(s.getBgImage(), x+offsetX, y + yPos);
                }
                return;
        }

    }

    /**
     * Calculates the starting drawing position for a certain dimension (horizontal/vertical) considering the given offset
     * 
     * @param offset The offset in the dimension
     * @param imageDim The image dimension (width/height)
     * @return The starting drawing position - 0 is the offset is 0 or a negative value if there is an offset
     */
    private int getTiledPosition(int offset,int imageDim) {
        if (offset!=0) {
            return offset%imageDim-imageDim;
        }
        return 0;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy