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

com.codename1.maven.CompileCSSMojo Maven / Gradle / Ivy

The newest version!
/*
 * To change this license header, choose License Headers in Project Properties.
 * To change this template file, choose Tools | Templates
 * and open the template in the editor.
 */
package com.codename1.maven;


import java.io.File;
import java.io.IOException;

import org.apache.commons.io.FileUtils;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.plugins.annotations.LifecyclePhase;
import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.plugins.annotations.ResolutionScope;
import org.apache.tools.ant.taskdefs.Copy;
import org.apache.tools.ant.taskdefs.Expand;
import org.apache.tools.ant.taskdefs.Java;
import org.apache.tools.ant.types.FileSet;

import static com.codename1.maven.PathUtil.path;

/**
 * Compiles the project's CSS files, generating a theme.res file which is placed in the build/classes directory.
 *
 * **Notes**
 *
 * . This mojo is only run for application projects.  Libary projects support CSS but they don't compile them.
 *   They just package them in a cn1css artifact which will be merged into the CSS for application projects that
 *   use them.
 * . The project must have the "codename1.cssTheme" property defined as "true" in either the pom.xml or the
 *   codenameone_settings.properties.  Otherwise this mojo does nothing.
 * . The CSS from cn1libs in dependencies is merged with the project CSS (located at src/main/css/theme.css) into
 *   a build directory at target/css.  This merged file is then compiled to the output file in build/classes/theme.res.
 *
 * @author shannah
 */
@Mojo(name = "css", defaultPhase = LifecyclePhase.PROCESS_RESOURCES, 
        requiresDependencyResolution = ResolutionScope.COMPILE_PLUS_RUNTIME,
        requiresDependencyCollection = ResolutionScope.COMPILE_PLUS_RUNTIME)
public class CompileCSSMojo extends AbstractCN1Mojo {


    @Override
    protected void executeImpl() throws MojoExecutionException, MojoFailureException {
        File cssDirectory = findCSSDirectory();
        int themeCssLen = "theme.css".length();
        if (cssDirectory != null && cssDirectory.isDirectory()) {
            for (File file : cssDirectory.listFiles()) {
                String fileName = file.getName();
                if (fileName.endsWith("theme.css")) {
                    executeImpl(fileName.substring(0, fileName.length() - themeCssLen));
                }
            }
        }
    }

    /**
     * Gets the source CSS directory (src/main/css).  The theme.css file should be inside this directory.
     * @return
     */
    protected File findCSSDirectory() {
        for (String dir : project.getCompileSourceRoots()) {
            File dirFile = new File(dir);
            File cssSibling = new File(dirFile.getParentFile(), "css");
            File themeCss = new File(cssSibling, "theme.css");
            if (themeCss.exists()) {
                return cssSibling;
            }
            
        }
        return null;
    }

    private void executeImpl(String themePrefix) throws MojoExecutionException, MojoFailureException {
        if (!isCN1ProjectDir()) {
            return;
        }
        if (properties.getProperty("codename1.cssTheme", null) == null) {
            getLog().info("CSS themes not activated for this project.  Skipping CSS compilation");
            return;
        }

        File cssDirectory = findCSSDirectory(); // src/main/css
        if (cssDirectory == null || !cssDirectory.exists()) {
            getLog().warn("CSS compilation skipped because no CSS theme was found");
            return;
        }
        File themeResOutput = new File(project.getBuild().getOutputDirectory() + File.separator + themePrefix + "theme.res");
        // target/css
        File cssBuildDir = new File(project.getBuild().getDirectory() + File.separator + "css");
        cssBuildDir.mkdirs();

        // target/css/theme.css - the merged CSS file
        File mergeFile = new File(cssBuildDir, themePrefix + "theme.css");
        mergeFile.getParentFile().mkdirs();
        try {
            if (themeResOutput.exists() && getCSSSourcesModificationTime() < themeResOutput.lastModified()) {
                getLog().info("CSS sources unchanged since last compile.  Skipping CSS compilation");
                return;
            }
        } catch (IOException ex) {
            throw new MojoExecutionException("Failed to check CSS file modification times", ex);
        }

        // Compile a comma-delimited list a CSS files that will be sent to the CSS compiler as inputs.
        // We look through all dependency artifacts with the cn1css classifier, and add their
        // theme.css to the input list.  (Codename One Library projects will include such an
        // artifact if they have CSS files).
        final StringBuilder inputs = new StringBuilder();

        project.getArtifacts().forEach(artifact->{
            if (artifact.hasClassifier() && "cn1css".equals(artifact.getClassifier())) {
                File zip = findArtifactFile(artifact);
                if (zip == null || !zip.exists()) {
                    return;
                }

                File extracted = new File(zip.getParentFile(), zip.getName()+"-extracted");
                getLog().debug("Checking for extracted CSS bundle "+extracted);
                if (extracted.exists() && artifact.isSnapshot() && getLastModified(artifact) > extracted.lastModified()) {
                    try {
                        FileUtils.deleteDirectory(extracted);
                    } catch (IOException ex){
                        getLog().error(ex);
                    }
                }
                if (!extracted.exists()) {
                    getLog().debug("CSS bundle "+zip+" not extracted yet.  Extracting to "+extracted);
                    // This is a cn1css artifact, which is a zip file.
                    // We extract it so that we can access the files directly.
                    Expand expand = (Expand)antProject.createTask("unzip");
                    expand.setSrc(zip);
                    expand.setDest(extracted);
                    expand.execute();

                }
                if (extracted.exists()) {
                    File extractedCssDir = new File(extracted, path("META-INF","codenameone", artifact.getGroupId(), artifact.getArtifactId(), "css"));
                    if (extractedCssDir.exists()) {
                        // We expect that the cn1css artifact has a theme.css file at its root
                        // If found, we add it to the list of inputs.
                        File theme = new File(extractedCssDir, themePrefix + "theme.css");
                        if (theme.exists()) {
                            if (inputs.length() > 0) {
                                inputs.append(",");
                            }
                            inputs.append(theme.getAbsolutePath());
                        }
                    }

                } else {
                    getLog().debug("CSS bundle extraction must have failed for "+zip+" because after extraction it still doesn't exist at "+extracted);
                }
            }
        });

        // The project's theme.css file is added to the input list last so that it will result in it
        // being last in the merged theme.css file (i.e. the application project CSS can override the
        // CSS in dependent libraries.
        File cssTheme = new File(cssDirectory, themePrefix + "theme.css");
        if (cssTheme.exists()) {
            if (inputs.length() > 0) {
                inputs.append(",");
            }
            inputs.append(cssTheme.getAbsolutePath());
        } else {
            if (themePrefix.isEmpty() && inputs.length() > 0) {
                throw new MojoFailureException("Cannot compile CSS for this project.  The project does not include a "+themePrefix+"-theme.css file in "+cssTheme+", but it includes dependencies that require CSS.  Please add a CSS file at "+cssTheme);

            }
            getLog().info("Skipping CSS compilation for because "+themePrefix + cssTheme+" does not exist");
            return;
        }



        // Run the CSS compiler which is contained inside the codenameone-designer jar
        // NOTE: The codenameone-designer.jar is a dependency of the codenameone-maven-plugin as
        // zip file (which is the designer jar with all dependencies).  We use this jar
        // rather than the central designer_1.jar located in the user's home directory to make it
        // easier to pin to a particular version.
        Java java = createJava();
        java.setDir(getCN1ProjectDir());
        java.setJar(getDesignerJar());
        java.setFork(true);
        java.setFailonerror(true);
        setupCef();
        String cefDir = System.getProperty("cef.dir", System.getProperty("user.home") + File.separator + ".codenameone" + File.separator + "cef");
        java.createJvmarg().setValue("-Dcli=true");
        java.createJvmarg().setValue("-Dcef.dir="+cefDir);
        java.createArg().setValue("-css");
        java.createArg().setValue("-input");
        java.createArg().setValue(inputs.toString());

        java.createArg().setValue("-output");
        java.createArg().setFile(themeResOutput);

        java.createArg().setValue("-merge");
        java.createArg().setFile(mergeFile);
        int res = java.executeJava();
        if (res != 0) {
            throw new MojoExecutionException("An error occurred while compiling the CSS files.  Inputs: "+inputs+", output: " + new File(project.getBuild().getOutputDirectory() + File.separator + themePrefix + "theme.res") +", merge file: "+mergeFile);
        }
    }
    
    
    
    
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy