You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
219 lines
7.0 KiB
219 lines
7.0 KiB
9 years ago
|
// Copyright (C) 2014 by Matthew York
|
||
|
//
|
||
|
// Permission is hereby granted, free of charge, to any
|
||
|
// person obtaining a copy of this software and
|
||
|
// associated documentation files (the "Software"), to
|
||
|
// deal in the Software without restriction, including
|
||
|
// without limitation the rights to use, copy, modify, merge,
|
||
|
// publish, distribute, sublicense, and/or sell copies of the
|
||
|
// Software, and to permit persons to whom the Software is
|
||
|
// furnished to do so, subject to the following conditions:
|
||
|
//
|
||
|
// The above copyright notice and this permission notice shall
|
||
|
// be included in all copies or substantial portions of the Software.
|
||
|
//
|
||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
||
|
// OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
||
|
// BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
||
|
// ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||
|
// CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||
|
|
||
|
#import "DTTimePeriodChain.h"
|
||
|
#import "DTError.h"
|
||
|
|
||
|
@interface DTTimePeriodChain ()
|
||
|
|
||
|
@end
|
||
|
|
||
|
@implementation DTTimePeriodChain
|
||
|
|
||
|
#pragma mark - Custom Init / Factory Chain
|
||
|
+(DTTimePeriodChain *)chain{
|
||
|
return [[DTTimePeriodChain alloc] init];
|
||
|
}
|
||
|
|
||
|
#pragma mark - Chain Existence Manipulation
|
||
|
-(void)addTimePeriod:(DTTimePeriod *)period{
|
||
|
if ([period class] != [DTTimePeriod class]) {
|
||
|
[DTError throwBadTypeException:period expectedClass:[DTTimePeriod class]];
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (periods) {
|
||
|
if (periods.count > 0) {
|
||
|
//Create a modified period to be added based on size of passed in period
|
||
|
DTTimePeriod *modifiedPeriod = [DTTimePeriod timePeriodWithSize:DTTimePeriodSizeSecond amount:period.durationInSeconds startingAt:[periods[periods.count - 1] EndDate]];
|
||
|
|
||
|
//Add object to periods array
|
||
|
[periods addObject:modifiedPeriod];
|
||
|
}
|
||
|
else {
|
||
|
//Add object to periods array
|
||
|
[periods addObject:period];
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
//Create new periods array
|
||
|
periods = [NSMutableArray array];
|
||
|
|
||
|
//Add object to periods array
|
||
|
[periods addObject:period];
|
||
|
}
|
||
|
|
||
|
//Set object's variables with updated array values
|
||
|
[self updateVariables];
|
||
|
}
|
||
|
|
||
|
-(void)insertTimePeriod:(DTTimePeriod *)period atInedx:(NSInteger)index{
|
||
|
if ([period class] != [DTTimePeriod class]) {
|
||
|
[DTError throwBadTypeException:period expectedClass:[DTTimePeriod class]];
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
//Make sure the index is within the operable bounds of the periods array
|
||
|
if (index == 0) {
|
||
|
//Update bounds of period to make it fit in chain
|
||
|
DTTimePeriod *modifiedPeriod = [DTTimePeriod timePeriodWithSize:DTTimePeriodSizeSecond amount:period.durationInSeconds endingAt:[periods[0] EndDate]];
|
||
|
|
||
|
//Insert the updated object at the beginning of the periods array
|
||
|
[periods insertObject:modifiedPeriod atIndex:0];
|
||
|
}
|
||
|
else if (index > 0 && index < periods.count) {
|
||
|
|
||
|
//Shift time periods later if they fall after new period
|
||
|
[periods enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
|
||
|
//Shift later
|
||
|
if (idx >= index) {
|
||
|
[((DTTimePeriod *) obj) shiftLaterWithSize:DTTimePeriodSizeSecond amount:period.durationInSeconds];
|
||
|
}
|
||
|
}];
|
||
|
|
||
|
//Update bounds of period to make it fit in chain
|
||
|
DTTimePeriod *modifiedPeriod = [DTTimePeriod timePeriodWithSize:DTTimePeriodSizeSecond amount:period.durationInSeconds startingAt:[periods[index - 1] EndDate]];
|
||
|
|
||
|
//Insert the updated object at the beginning of the periods array
|
||
|
[periods insertObject:modifiedPeriod atIndex:index];
|
||
|
|
||
|
//Set object's variables with updated array values
|
||
|
[self updateVariables];
|
||
|
}
|
||
|
else {
|
||
|
[DTError throwInsertOutOfBoundsException:index array:periods];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
-(void)removeTimePeriodAtIndex:(NSInteger)index{
|
||
|
//Make sure the index is within the operable bounds of the periods array
|
||
|
if (index >= 0 && index < periods.count) {
|
||
|
DTTimePeriod *period = periods[index];
|
||
|
|
||
|
//Shift time periods later if they fall after new period
|
||
|
[periods enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
|
||
|
//Shift earlier
|
||
|
if (idx > index) {
|
||
|
[((DTTimePeriod *) obj) shiftEarlierWithSize:DTTimePeriodSizeSecond amount:period.durationInSeconds];
|
||
|
}
|
||
|
}];
|
||
|
|
||
|
//Remove object
|
||
|
[periods removeObjectAtIndex:index];
|
||
|
|
||
|
//Set object's variables with updated array values
|
||
|
[self updateVariables];
|
||
|
}
|
||
|
else {
|
||
|
[DTError throwRemoveOutOfBoundsException:index array:periods];
|
||
|
}
|
||
|
}
|
||
|
-(void)removeLatestTimePeriod{
|
||
|
if (periods.count > 0) {
|
||
|
[periods removeLastObject];
|
||
|
|
||
|
//Update the object variables
|
||
|
if (periods.count > 0) {
|
||
|
//Set object's variables with updated array values
|
||
|
[self updateVariables];
|
||
|
}
|
||
|
else {
|
||
|
[self setVariablesNil];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
-(void)removeEarliestTimePeriod{
|
||
|
if (periods > 0) {
|
||
|
//Shift time periods earlier
|
||
|
[periods enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
|
||
|
//Shift earlier to account for removal of first element in periods array
|
||
|
[((DTTimePeriod *) obj) shiftEarlierWithSize:DTTimePeriodSizeSecond amount:[periods[0] durationInSeconds]];
|
||
|
}];
|
||
|
|
||
|
//Remove first period
|
||
|
[periods removeObjectAtIndex:0];
|
||
|
|
||
|
//Update the object variables
|
||
|
if (periods.count > 0) {
|
||
|
//Set object's variables with updated array values
|
||
|
[self updateVariables];
|
||
|
}
|
||
|
else {
|
||
|
[self setVariablesNil];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#pragma mark - Chain Relationship
|
||
|
-(BOOL)isEqualToChain:(DTTimePeriodChain *)chain{
|
||
|
//Check class
|
||
|
if ([chain class] != [DTTimePeriodChain class]) {
|
||
|
[DTError throwBadTypeException:chain expectedClass:[DTTimePeriodChain class]];
|
||
|
return NO;
|
||
|
}
|
||
|
|
||
|
//Check group level characteristics for speed
|
||
|
if (![self hasSameCharacteristicsAs:chain]) {
|
||
|
return NO;
|
||
|
}
|
||
|
|
||
|
//Check whole chain
|
||
|
__block BOOL isEqual = YES;
|
||
|
[periods enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
|
||
|
if (![chain[idx] isEqualToPeriod:obj]) {
|
||
|
isEqual = NO;
|
||
|
*stop = YES;
|
||
|
}
|
||
|
}];
|
||
|
return isEqual;
|
||
|
}
|
||
|
|
||
|
#pragma mark - Getters
|
||
|
|
||
|
-(DTTimePeriod *)First{
|
||
|
return First;
|
||
|
}
|
||
|
|
||
|
-(DTTimePeriod *)Last{
|
||
|
return Last;
|
||
|
}
|
||
|
|
||
|
#pragma mark - Helper Methods
|
||
|
|
||
|
-(void)updateVariables{
|
||
|
//Set helper variables
|
||
|
StartDate = [periods[0] StartDate];
|
||
|
EndDate = [periods[periods.count - 1] EndDate];
|
||
|
First = periods[0];
|
||
|
Last = periods[periods.count -1];
|
||
|
}
|
||
|
|
||
|
-(void)setVariablesNil{
|
||
|
//Set helper variables
|
||
|
StartDate = nil;
|
||
|
EndDate = nil;
|
||
|
First = nil;
|
||
|
Last = nil;
|
||
|
}
|
||
|
|
||
|
@end
|