| /* |
| * Copyright (C) 2008 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| import java.io.File; |
| import java.io.IOException; |
| import java.util.SortedSet; |
| import java.util.TreeMap; |
| import java.util.TreeSet; |
| import java.util.Collection; |
| import java.util.ArrayList; |
| import java.util.List; |
| import java.util.regex.Pattern; |
| |
| /** |
| * Generates an Eclipse project. |
| */ |
| public class Eclipse { |
| |
| /** |
| * Generates an Eclipse .classpath file from the given configuration. |
| */ |
| public static void generateFrom(Configuration c) throws IOException { |
| StringBuilder classpath = new StringBuilder(); |
| |
| classpath.append("<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n" |
| + "<classpath>\n"); |
| |
| /* |
| * If the user has a file named "path-precedence" in their project's |
| * root directory, we'll order source roots based on how they match |
| * regular expressions in that file. Source roots that match earlier |
| * patterns will come sooner in configuration file. |
| */ |
| List<Pattern> patterns = new ArrayList<Pattern>(); |
| |
| File precedence = new File("path-precedence"); |
| if (precedence.exists()) { |
| Configuration.parseFile(precedence, patterns); |
| } else { |
| // Put ./out at the bottom by default. |
| patterns.add(Pattern.compile("^(?!out/)")); |
| } |
| |
| // Everything not matched by the user's precedence spec. |
| patterns.add(Pattern.compile(".*")); |
| |
| |
| List<Bucket> buckets = new ArrayList<Bucket>(patterns.size()); |
| for (Pattern pattern : patterns) { |
| buckets.add(new Bucket(pattern)); |
| } |
| |
| // Put source roots in respective buckets. |
| OUTER: for (File sourceRoot : c.sourceRoots) { |
| // Trim preceding "./" from path. |
| String path = sourceRoot.getPath().substring(2); |
| |
| for (Bucket bucket : buckets) { |
| if (bucket.matches(path)) { |
| bucket.sourceRoots.add(sourceRoot); |
| continue OUTER; |
| } |
| } |
| } |
| |
| // Output source roots to configuration file. |
| for (Bucket bucket : buckets) { |
| for (File sourceRoot : bucket.sourceRoots) { |
| classpath.append(" <classpathentry kind=\"src\""); |
| CharSequence excluding = constructExcluding(sourceRoot, c); |
| if (excluding.length() > 0) { |
| classpath.append(" excluding=\"") |
| .append(excluding).append("\""); |
| } |
| classpath.append(" path=\"") |
| .append(trimmed(sourceRoot)).append("\"/>\n"); |
| } |
| |
| } |
| |
| // Output .jar entries. |
| for (File jar : c.jarFiles) { |
| classpath.append(" <classpathentry kind=\"lib\" path=\"") |
| .append(trimmed(jar)).append("\"/>\n"); |
| } |
| |
| /* |
| * Output directory. Unfortunately, Eclipse forces us to put it |
| * somewhere under the project directory. |
| */ |
| classpath.append(" <classpathentry kind=\"output\" path=\"" |
| + "out/eclipse\"/>\n"); |
| |
| classpath.append("</classpath>\n"); |
| |
| Files.toFile(classpath.toString(), new File(".classpath")); |
| } |
| |
| |
| /** |
| * Constructs the "excluding" argument for a given source root. |
| */ |
| private static CharSequence constructExcluding(File sourceRoot, |
| Configuration c) { |
| StringBuilder classpath = new StringBuilder(); |
| String path = sourceRoot.getPath(); |
| |
| // Exclude nested source roots. |
| SortedSet<File> nextRoots = c.sourceRoots.tailSet(sourceRoot); |
| int count = 0; |
| for (File nextRoot : nextRoots) { |
| // The first root is this root. |
| if (count == 0) { |
| count++; |
| continue; |
| } |
| |
| String nextPath = nextRoot.getPath(); |
| if (!nextPath.startsWith(path)) { |
| break; |
| } |
| |
| if (count > 1) { |
| classpath.append('|'); |
| } |
| classpath.append(nextPath.substring(path.length() + 1)) |
| .append('/'); |
| |
| count++; |
| } |
| |
| // Exclude excluded directories under this source root. |
| SortedSet<File> excludedDirs = c.excludedDirs.tailSet(sourceRoot); |
| for (File excludedDir : excludedDirs) { |
| String excludedPath = excludedDir.getPath(); |
| if (!excludedPath.startsWith(path)) { |
| break; |
| } |
| |
| if (count > 1) { |
| classpath.append('|'); |
| } |
| classpath.append(excludedPath.substring(path.length() + 1)) |
| .append('/'); |
| |
| count++; |
| } |
| |
| return classpath; |
| } |
| |
| /** |
| * Returns the trimmed path. |
| */ |
| private static String trimmed(File file) { |
| return file.getPath().substring(2); |
| } |
| |
| /** |
| * A precedence bucket for source roots. |
| */ |
| private static class Bucket { |
| |
| private final Pattern pattern; |
| private final List<File> sourceRoots = new ArrayList<File>(); |
| |
| private Bucket(Pattern pattern) { |
| this.pattern = pattern; |
| } |
| |
| private boolean matches(String path) { |
| return pattern.matcher(path).find(); |
| } |
| } |
| } |