// [The "BSD licence"]
// Copyright (c) 2006-2007 Kay Roepke 2010 Alan Condit
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions
// are met:
// 1. Redistributions of source code must retain the above copyright
//    notice, this list of conditions and the following disclaimer.
// 2. Redistributions in binary form must reproduce the above copyright
//    notice, this list of conditions and the following disclaimer in the
//    documentation and/or other materials provided with the distribution.
// 3. The name of the author may not be used to endorse or promote products
//    derived from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
// IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
// OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
// IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
// NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
// THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

#import "CommonTreeAdaptor.h"

@implementation CommonTreeAdaptor

+ (CommonTree *) newEmptyTree;
{
    return [CommonTree newTree];
}

+ (CommonTreeAdaptor *)newTreeAdaptor
{
    return[[CommonTreeAdaptor alloc] init];
}

- (id) init
{
    self = [super init];
    if (self) {
    }
    return self;
}

/** Duplicate a node.  This is part of the factory;
 *	override if you want another kind of node to be built.
 *
 *  I could use reflection to prevent having to override this
 *  but reflection is slow.
 */
- (id) dupNode:(id<BaseTree>)t
{
    if ( t==nil )
        return nil;
    return [CommonTree newTree:t];
}

/** Tell me how to create a token for use with imaginary token nodes.
 *  For example, there is probably no input symbol associated with imaginary
 *  token DECL, but you need to create it as a payload or whatever for
 *  the DECL node as in ^(DECL type ID).
 *
 *  This is a variant of createToken where the new token is derived from
 *  an actual real input token.  Typically this is for converting '{'
 *  tokens to BLOCK etc...  You'll see
 *
 *    r : lc='{' ID+ '}' -> ^(BLOCK[$lc] ID+) ;
 *
 *  If you care what the token payload objects' type is, you should
 *  override this method and any other createToken variant.
 */
- (CommonTree *) create:(CommonToken *)aToken
{
    return [CommonTree newTreeWithToken:aToken];
}

/** Tell me how to create a token for use with imaginary token nodes.
 *  For example, there is probably no input symbol associated with imaginary
 *  token DECL, but you need to create it as a payload or whatever for
 *  the DECL node as in ^(DECL type ID).
 *
 *  If you care what the token payload objects' type is, you should
 *  override this method and any other createToken variant.
 */
- (CommonTree *)createTree:(NSInteger)tokenType Text:(NSString *)text
{
    return [CommonTree newTreeWithTokenType:tokenType Text:text];
}

- (id<Token>)createToken:(NSInteger)tokenType Text:(NSString *)text
{
    id<Token> fromToken = [CommonToken newToken:tokenType Text:text];
    return fromToken;
}

- (id<Token>)createToken:(id<Token>)fromToken
{
    return [CommonToken newTokenWithToken:(CommonToken *)fromToken];
}

/** Track start/stop token for subtree root created for a rule.
 *  Only works with Tree nodes.  For rules that match nothing,
 *  seems like this will yield start=i and stop=i-1 in a nil node.
 *  Might be useful info so I'll not force to be i..i.
 */
- (void) setTokenBoundaries:(id<BaseTree>)aTree From:(id<Token>)startToken To:(id<Token>)stopToken
{
    if ( aTree == nil )
        return;
    int startTokIdx = 0;
    int stopTokIdx = 0;
    if ( startToken != nil )
        startTokIdx = [startToken getTokenIndex];
    if ( stopToken != nil )
        stopTokIdx = [stopToken getTokenIndex];
    [(id<BaseTree>)aTree setTokenStartIndex:startTokIdx];
    [(id<BaseTree>)aTree setTokenStopIndex:stopTokIdx];
}

- (NSInteger)getTokenStartIndex:(id<BaseTree>) t
{
    if ( t == nil )
        return -1;
    return [(id<BaseTree>)t getTokenStartIndex];
}

- (NSInteger)getTokenStopIndex:(id<BaseTree>) t
{
    if ( t == nil )
        return -1;
    return [(id<BaseTree>)t getTokenStopIndex];
}

- (NSString *)getText:(CommonTree *)t
{
    if ( t == nil )
        return nil;
    return t.token.text;
}

- (void)setText:(id<BaseTree>)t Text:(NSString *)text
{
    if ( t == nil )
        return;
}

- (NSInteger)getType:(CommonTree *)t
{
    if ( t==nil )
        return TokenTypeInvalid;
    return t.token.type;
}

- (void) setType:(id<BaseTree>)t Type:(NSInteger)tokenType
{
    if ( t==nil )
        return;
}

/** What is the Token associated with this node?  If
 *  you are not using CommonTree, then you must
 *  override this in your own adaptor.
 */
- (id<Token>) getToken:(CommonTree *) t
{
    if ( [t isKindOfClass:[CommonTree class]] ) {
        return t.token;
    }
    return nil; // no idea what to do
}

- (id<BaseTree>) getChild:(id<BaseTree>)t At:(NSInteger)i
{
    if ( t == nil )
        return nil;
    return [(id<BaseTree>)t getChild:i];
}

- (void) setChild:(id<BaseTree>)t At:(NSInteger)i Child:(id<BaseTree>)child
{
    if ( t == nil )
        return;
    [(id<BaseTree>)t setChild:i With:child];
}

- (id) deleteChild:(id<BaseTree>)t Index:(NSInteger)anIndex
{
    return [t deleteChild:anIndex];
}

- (NSInteger) getChildCount:(id<BaseTree>) t
{
    if ( t == nil )
        return 0;
    return [(id<BaseTree>) t getChildCount];
}

- (id<BaseTree>) getParent:(id<BaseTree>) t
{
    if ( t == nil )
        return nil;
    return (id<BaseTree>)[t getParent];
}

- (void) setParent:(id<BaseTree>)t With:(id<BaseTree>) parent
{
    if ( t != nil )
        [(id<BaseTree>) t setParent:(id<BaseTree>)parent];
}

- (NSInteger) getChildIndex:(id<BaseTree>) t
{
    if ( t == nil )
        return 0;
    return [(id<BaseTree>) t getChildIndex];
}

- (void) setChildIndex:(id<BaseTree>)t With:(NSInteger)anIndex
{
    if ( t!=nil )
        [(id<BaseTree>)t setChildIndex:anIndex];
}

- (void) replaceChildren:(id<BaseTree>)parent From:(NSInteger)startChildIndex To:(NSInteger)stopChildIndex With:(id<BaseTree>)t
{
    if ( parent != nil ) {
        [(id<BaseTree>)parent replaceChildrenFrom:startChildIndex To:stopChildIndex With:t];
    }
}

- (id) copyWithZone:(NSZone *)aZone
{
    return [[[self class] allocWithZone:aZone] init];
}

@end