/*********************************************************************
Copyright 2014 the Flapi authors
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.
********************************************************************/
package unquietcode.tools.flapi.helpers;
import unquietcode.tools.flapi.DescriptorBuilderException;
import unquietcode.tools.flapi.annotations.AnnotationIntrospector;
import unquietcode.tools.flapi.builder.Block.BlockHelper;
import unquietcode.tools.flapi.builder.Method.MethodHelper;
import unquietcode.tools.flapi.outline.BlockOutline;
import unquietcode.tools.flapi.outline.BlockReference;
import unquietcode.tools.flapi.outline.MethodOutline;
import java.util.concurrent.atomic.AtomicReference;
/**
* @author Ben Fagin
* @version 03-04-2012
*/
public class BlockHelperImpl implements BlockHelper {
final BlockOutline block;
public BlockHelperImpl(BlockOutline block) {
this.block = block;
}
@Override
public void addMethod(String methodSignature, AtomicReference<MethodHelper> _helper1) {
_addMethod(block, methodSignature, _helper1);
}
static void _addMethod(BlockOutline block, String methodSignature, AtomicReference<MethodHelper> _helper1) {
MethodOutline m = block.addMethod(methodSignature);
_helper1.set(new MethodHelperImpl(m));
}
@Override
public void startBlock(String blockName, String methodSignature, AtomicReference<MethodHelper> _helper1, AtomicReference<BlockHelper> _helper2) {
_startBlock(block, blockName, methodSignature, _helper1, _helper2);
}
static void _startBlock(BlockOutline block, String blockName, String methodSignature, AtomicReference<MethodHelper> _helper1, AtomicReference<BlockHelper> _helper2) {
BlockOutline newBlock = block.addBlock(blockName);
MethodOutline blockMethod = block.addMethod(methodSignature);
blockMethod.getBlockChain().add(newBlock);
newBlock.setConstructor(blockMethod);
_helper1.set(new MethodHelperImpl(blockMethod));
_helper2.set(new BlockHelperImpl(newBlock));
}
@Override
public void startBlock(String methodSignature, AtomicReference<MethodHelper> _helper1, AtomicReference<BlockHelper> _helper2) {
_startBlock(block, null, methodSignature, _helper1, _helper2);
}
@Override
public void addBlockReference(String blockName, String methodSignature, AtomicReference<MethodHelper> _helper1) {
_addBlockReference(block, blockName, methodSignature, _helper1);
}
static void _addBlockReference(
BlockOutline block,
String blockName,
String methodSignature,
AtomicReference<MethodHelper> _helper1
){
final BlockReference blockReference;
// Allow referencing the current block, and just
// let the resolution logic do its work later on.
if ("this".equals(blockName)) {
blockReference = new BlockReference(block);
} else {
blockReference = new BlockReference(blockName);
}
MethodOutline blockMethod = block.addMethod(methodSignature);
blockMethod.getBlockChain().add(blockReference);
blockReference.setConstructor(blockMethod);
_helper1.set(new MethodHelperImpl(blockMethod));
}
@Override
public void addEnumSelector(Class<?> clazz, String methodSignature, AtomicReference<MethodHelper> _helper1) {
_addEnumSelector(block, clazz, methodSignature, _helper1);
}
static void _addEnumSelector(BlockOutline block, Class<?> clazz, String methodSignature, AtomicReference<MethodHelper> _helper1) {
if (clazz == null) {
throw new NullPointerException("addEnumSelector: class is null");
} else if (!clazz.isEnum()) {
throw new DescriptorBuilderException("addEnumSelector: class must be an enum class");
}
AtomicReference<BlockHelper> blockHelper = new AtomicReference<BlockHelper>();
_startBlock(block, clazz.getSimpleName(), methodSignature, _helper1, blockHelper);
for (Object value : clazz.getEnumConstants()) {
String name = ((Enum) value).name();
// for every enum value, add a new terminal method
AtomicReference<MethodHelper> methodHelper = new AtomicReference<MethodHelper>();
blockHelper.get().addMethod(name+"()", methodHelper);
methodHelper.get().last();
}
}
@Override
public void addMixin(Class helper) {
_addMixin(block, helper);
}
static void _addMixin(BlockOutline block, Class mixin) {
if (!AnnotationIntrospector.isAnnotated(mixin)) {
System.err.println("the provided mixin type contains no Flapi method annotations");
}
block.addMixin(mixin);
}
@Override
public void addMixin(String blockName) {
_addMixin(block, blockName);
}
static void _addMixin(BlockOutline block, String mixin) {
block.addMixin(mixin);
}
@Override
public void endBlock() {
// nothing
}
}