001/* 002 * BioJava development code 003 * 004 * This code may be freely distributed and modified under the 005 * terms of the GNU Lesser General Public Licence. This should 006 * be distributed with the code. If you do not have a copy, 007 * see: 008 * 009 * http://www.gnu.org/copyleft/lesser.html 010 * 011 * Copyright for this code is held jointly by the individual 012 * authors. These should be listed in @author doc comments. 013 * 014 * For more information on the BioJava project and its aims, 015 * or to join the biojava-l mailing list, visit the home page 016 * at: 017 * 018 * http://www.biojava.org/ 019 * 020 */ 021 022package org.biojava.nbio.structure.symmetry.core; 023 024import java.util.ArrayList; 025import java.util.HashSet; 026import java.util.Iterator; 027import java.util.List; 028import java.util.Set; 029 030/** 031 * 032 * @author Peter 033 */ 034public class PermutationGroup implements Iterable<List<Integer>> { 035 List<List<Integer>> permutations = new ArrayList<>(); 036 037 public void addPermutation(List<Integer> permutation) { 038 if (!permutations.contains(permutation)) { 039 permutations.add(permutation); 040 } 041 } 042 043 public List<Integer> getPermutation(int index) { 044 return permutations.get(index); 045 } 046 047 public int getOrder() { 048 return permutations.size(); 049 } 050 051 052 /** 053 * Starts with an incomplete set of group generators in `permutations` and 054 * expands it to include all possible combinations. 055 * 056 * Ways to complete group: 057 * - combinations of permutations pi x pj 058 * - combinations with itself p^k 059 * 060 */ 061 public void completeGroup() { 062 // Copy initial set to allow permutations to grow 063 List<List<Integer>> gens = new ArrayList<>(permutations); 064 // Keep HashSet version of permutations for fast lookup. 065 Set<List<Integer>> known = new HashSet<>(permutations); 066 //breadth-first search through the map of all members 067 List<List<Integer>> currentLevel = new ArrayList<>(permutations); 068 while( currentLevel.size() > 0) { 069 List<List<Integer>> nextLevel = new ArrayList<>(); 070 for( List<Integer> p : currentLevel) { 071 for(List<Integer> gen : gens) { 072 List<Integer> y = combine(p,gen); 073 if(!known.contains(y)) { 074 nextLevel.add(y); 075 //bypass addPermutation(y) for performance 076 permutations.add(y); 077 known.add(y); 078 } 079 } 080 } 081 currentLevel = nextLevel; 082 } 083 } 084 085 @Override 086 public String toString() { 087 StringBuilder sb = new StringBuilder(); 088 sb.append("Permutation Group: " + permutations.size() + " permutation"); 089 for (List<Integer> permutation : permutations) { 090 sb.append(permutation.toString()); 091 } 092 return sb.toString(); 093 } 094 095 public static List<Integer> combine(List<Integer> permutation1, List<Integer> permutation2) { 096 List<Integer> intermediate = new ArrayList<>(permutation1.size()); 097 for (int i = 0, n = permutation1.size(); i < n; i++) { 098 intermediate.add(permutation2.get(permutation1.get(i))); 099 } 100 return intermediate; 101 } 102 103 public static int getOrder(List<Integer> permutation) { 104 List<Integer> copy = new ArrayList<>(permutation); 105 for (int i = 0, n = permutation.size(); i < n; i++) { 106 copy = combine(copy, permutation); 107 if (copy.equals(permutation)) { 108 return i + 1; 109 } 110 } 111 return 0; 112 } 113 114 public String getGroupTable() { 115 StringBuilder builder = new StringBuilder(); 116 builder.append(" |"); 117 for (int i = 0; i < getOrder(); i++) { 118 builder.append(" "); 119 builder.append(i); 120 } 121 builder.append("\n"); 122 builder.append("---"); 123 for (int i = 0; i < getOrder(); i++) { 124 builder.append("--"); 125 } 126 builder.append("\n"); 127 for (int i = 0; i < getOrder(); i++) { 128 builder.append(i); 129 builder.append(" |"); 130 for (int j = 0; j < getOrder(); j++) { 131 builder.append(" "); 132 builder.append(permutations.indexOf(combine(permutations.get(i), permutations.get(j)))); 133 } 134 builder.append("\n"); 135 } 136 return builder.toString(); 137 } 138 139 @Override 140 public int hashCode() { 141 return getGroupTable().hashCode(); 142 } 143 144 @Override 145 public Iterator<List<Integer>> iterator() { 146 return permutations.iterator(); 147 } 148 149 @Override 150 public boolean equals(Object obj) { 151 if (this == obj) { 152 return true; 153 } 154 if (obj == null) { 155 return false; 156 } 157 if (this.getClass() == obj.getClass()) { 158 return permutations.equals(((PermutationGroup)obj).permutations); 159 } 160 return false; 161 } 162 163}