package tim.prune.data; import java.util.List; import tim.prune.UpdateMessageBroker; /** * Class to hold all track information, including data * and the selection information */ public class TrackInfo { private UpdateMessageBroker _broker = null; private Track _track = null; private Selection _selection = null; private FileInfo _fileInfo = null; private PhotoList _photoList = null; /** * Constructor * @param inTrack Track object * @param inBroker broker object */ public TrackInfo(Track inTrack, UpdateMessageBroker inBroker) { _broker = inBroker; _track = inTrack; _selection = new Selection(_track, inBroker); _fileInfo = new FileInfo(); _photoList = new PhotoList(); } /** * @return the Track object */ public Track getTrack() { return _track; } /** * @return the Selection object */ public Selection getSelection() { return _selection; } /** * @return the FileInfo object */ public FileInfo getFileInfo() { return _fileInfo; } /** * @return the PhotoList object */ public PhotoList getPhotoList() { return _photoList; } /** * Get the currently selected point, if any * @return DataPoint if single point selected, otherwise null */ public DataPoint getCurrentPoint() { return _track.getPoint(_selection.getCurrentPointIndex()); } /** * Get the currently selected photo, if any * @return Photo if selected, otherwise null */ public Photo getCurrentPhoto() { return _photoList.getPhoto(_selection.getCurrentPhotoIndex()); } /** * Load the specified data into the Track * @param inFieldArray array of Field objects describing fields * @param inPointArray 2d object array containing data * @param inAltFormat altitude format */ public void loadTrack(Field[] inFieldArray, Object[][] inPointArray, int inAltFormat) { _track.cropTo(0); _track.load(inFieldArray, inPointArray, inAltFormat); _selection.clearAll(); } /** * Add a List of Photos * @param inList List containing Photo objects * @return array containing number of photos and number of points added */ public int[] addPhotos(List inList) { // TODO: Should photos be sorted at load-time, either by filename or date? // Firstly count number of points and photos to add int numPhotosToAdd = 0; int numPointsToAdd = 0; if (inList != null && !inList.isEmpty()) { for (int i=0; i 0) { DataPoint[] dataPoints = new DataPoint[numPointsToAdd]; int pointNum = 0; boolean hasAltitude = false; // Add each Photo in turn for (int i=0; i 0) { // add points to track _track.appendPoints(dataPoints); // modify track field list _track.getFieldList().extendList(Field.LATITUDE); _track.getFieldList().extendList(Field.LONGITUDE); if (hasAltitude) {_track.getFieldList().extendList(Field.ALTITUDE);} } } int[] result = {numPhotosToAdd, numPointsToAdd}; return result; } /** * Delete the currently selected range of points * @return true if successful */ public boolean deleteRange() { int startSel = _selection.getStart(); int endSel = _selection.getEnd(); boolean answer = _track.deleteRange(startSel, endSel); // clear range selection _selection.modifyRangeDeleted(); return answer; } /** * Delete the currently selected point * @return true if point deleted */ public boolean deletePoint() { if (_track.deletePoint(_selection.getCurrentPointIndex())) { _selection.modifyPointDeleted(); _broker.informSubscribers(); return true; } return false; } /** * Delete the currently selected photo and optionally its point too * @param inPointToo true to also delete associated point * @return true if delete successful */ public boolean deleteCurrentPhoto(boolean inPointToo) { // delete currently selected photo int photoIndex = _selection.getCurrentPhotoIndex(); if (photoIndex >= 0) { Photo photo = _photoList.getPhoto(photoIndex); _photoList.deletePhoto(photoIndex); // has it got a point? if (photo.getDataPoint() != null) { if (inPointToo) { // delete point int pointIndex = _track.getPointIndex(photo.getDataPoint()); _track.deletePoint(pointIndex); } else { // disconnect point from photo photo.getDataPoint().setPhoto(null); photo.setDataPoint(null); } } // update subscribers _selection.modifyPointDeleted(); _broker.informSubscribers(); } return true; } /** * Compress the track to the given resolution * @param inResolution resolution * @return number of points deleted */ public int compress(int inResolution) { int numDeleted = _track.compress(inResolution); if (numDeleted > 0) _selection.clearAll(); return numDeleted; } /** * Delete all the duplicate points in the track * @return number of points deleted */ public int deleteDuplicates() { int numDeleted = _track.deleteDuplicates(); if (numDeleted > 0) _selection.clearAll(); return numDeleted; } /** * Clone the selected range of data points * @return shallow copy of DataPoint objects */ public DataPoint[] cloneSelectedRange() { return _track.cloneRange(_selection.getStart(), _selection.getEnd()); } /** * Interpolate extra points between two selected ones * @param inStartIndex start index of interpolation * @param inNumPoints num points to insert * @return true if successful */ public boolean interpolate(int inNumPoints) { boolean success = _track.interpolate(_selection.getStart(), inNumPoints); if (success) _selection.selectRangeEnd(_selection.getEnd() + inNumPoints); return success; } /** * Select the given DataPoint * @param inPoint DataPoint object to select */ public void selectPoint(DataPoint inPoint) { // get the index of the given Point int index = _track.getPointIndex(inPoint); // give to selection _selection.selectPoint(index); } /** * Select the given Photo and its point if any * @param inPhotoIndex index of photo to select */ public void selectPhoto(int inPhotoIndex) { // Find Photo object Photo photo = _photoList.getPhoto(inPhotoIndex); if (photo != null) { // Find point object and its index int pointIndex = _track.getPointIndex(photo.getDataPoint()); // give to selection object _selection.selectPhotoAndPoint(inPhotoIndex, pointIndex); } else { // no photo, just reset selection _selection.selectPhotoAndPoint(-1, -1); } } /** * Fire a trigger to all data subscribers */ public void triggerUpdate() { _broker.informSubscribers(); } }