/*******************************************************************************
* Copyright 2013 Naver Business Platform Corp.
*
* 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 com.handmark.pulltorefresh.library;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import android.content.Context;
import android.util.Log;
import android.view.View;
import com.handmark.pulltorefresh.configuration.xml.PullToRefreshXmlConfiguration;
import com.handmark.pulltorefresh.library.PullToRefreshBase.Mode;
import com.handmark.pulltorefresh.library.internal.DefaultIndicatorLayout;
import com.handmark.pulltorefresh.library.internal.IndicatorLayout;
/**
* Factory which creates indicator layouts
* <br />Indicator layouts must be listed in pulltorefresh.xml as "PullToRefresh/IndicatorLayouts/layout" nodes
* @author Wonjun Kim
*/
public class IndicatorLayoutFactory {
private static final String LOG_TAG = IndicatorLayoutFactory.class.getName();
/**
* Create the class token matched by <b>{@code layoutCode}</b>
* @param layoutCode Indicator layout code, which must be defined in pulltorefresh.xml
* @return Class token which is matched by {@code layoutCode}, or the class token of {@code DefaultIndicatorLayout} instance if not
*/
public static Class<? extends IndicatorLayout> createIndicatorLayoutClazzByLayoutCode(String layoutCode) {
String clazzName = PullToRefreshXmlConfiguration.getInstance().getIndicatorLayoutClazzName(layoutCode);
return createIndicatorLayoutClazz(clazzName);
}
/**
* Create the class token matched by <b>class name</b>
* @param clazzName Class name such as "com.handmark.pulltorefresh.library.internal.DefaultIndicatorLayout"
* @return Class token if the class matched by class name exists, or the class token of {@code DefaultIndicatorLayout} instance if not
*/
@SuppressWarnings("unchecked")
public static Class<? extends IndicatorLayout> createIndicatorLayoutClazz(String clazzName) {
Class<? extends IndicatorLayout> clazz = null;
if (clazzName == null) {
clazz = DefaultIndicatorLayoutFactory.createIndicatorLayoutClazz(clazzName);
return clazz;
}
try {
clazz = (Class<? extends IndicatorLayout> )Class.forName(clazzName);
} catch (ClassNotFoundException e) {
Log.e(LOG_TAG, "The indicator layout you have chosen class has not been found.", e);
clazz = DefaultIndicatorLayoutFactory.createIndicatorLayoutClazz(clazzName);
}
return clazz;
}
/**
* Create a {@code IndicatorLayout} instance matched by <b>{@code layoutCode}</b>
* @param layoutCode Indicator layout code, which must be defined in pulltorefresh.xml
* @param context
* @param mode
* @return {@code IndicatorLayout} instance if the class matched by {@code layoutCode} exists, or {@code DefaultIndicatorLayout} instance if not
*/
public static IndicatorLayout createIndicatorLayout(String layoutCode, Context context, PullToRefreshBase.Mode mode) {
Class<? extends IndicatorLayout> clazz = createIndicatorLayoutClazz(layoutCode);
return createIndicatorLayout(clazz, context, mode);
}
/**
* Create a {@code IndicatorLayout} instance matched by <b>{@code clazz} token</b>
* @param clazz Indicator layout class token, which must be defined in pulltorefresh.xml
* @param context
* @param mode
* @return {@code IndicatorLayout} instance if the class matched by {@code layoutCode} exists, or {@code DefaultIndicatorLayout} instance if not
*/
public static IndicatorLayout createIndicatorLayout(
Class<? extends IndicatorLayout> clazz, Context context, Mode mode) {
IndicatorLayout layout = null;
// Prevent NullPointerException
if ( clazz == null ) {
Log.i(LOG_TAG, "The Class token of the Indicator Layout is missing. Default Indicator Layout will be used.");
clazz = DefaultIndicatorLayoutFactory.createIndicatorLayoutClazz("");
}
layout = tryNewInstance(clazz, context, mode);
// If trying to create new instance has failed,
if (layout == null) {
layout = DefaultIndicatorLayoutFactory.createIndicatorLayout(clazz, context, mode);
}
layout.setVisibility(View.INVISIBLE);
return layout;
}
private static IndicatorLayout tryNewInstance(
Class<? extends IndicatorLayout> clazz, Context context, Mode mode) {
IndicatorLayout layout = null;
try {
Constructor<? extends IndicatorLayout> constructor = clazz
.getConstructor(Context.class, Mode.class);
layout = (IndicatorLayout) constructor.newInstance(context, mode);
} catch (IllegalArgumentException e) {
Log.e(LOG_TAG, "The indicator layout has failed to be created. ", e);
} catch (InvocationTargetException e) {
Log.e(LOG_TAG, "The indicator layout has failed to be created. ", e);
} catch (SecurityException e) {
Log.e(LOG_TAG, "The indicator layout has failed to be created. ", e);
} catch (NoSuchMethodException e) {
Log.e(LOG_TAG, "The indicator layout has failed to be created. ", e);
} catch (InstantiationException e) {
Log.e(LOG_TAG, "The indicator layout has failed to be created. ", e);
} catch (IllegalAccessException e) {
Log.e(LOG_TAG, "The indicator layout has failed to be created. ", e);
} catch (NullPointerException e) {
Log.e(LOG_TAG, "The indicator layout has failed to be created. ", e);
}
return layout;
}
/**
* Factory which creates a default indicator layout instance. This is used when {@code IndicatorLayoutFactory} fails to create a instance
* @author Wonjun Kim
*
*/
private static class DefaultIndicatorLayoutFactory {
/**
* @param clazzName This class name is being ignored
* @return Class token of {@code DefaultIndicatorLayout}
*/
public static Class<? extends IndicatorLayout> createIndicatorLayoutClazz(String clazzName) {
return DefaultIndicatorLayout.class;
}
/**
* @param clazz Class token is being ignored.
* @param context
* @param mode
* @return {@code DefaultIndicatorLayout} instance
*/
public static IndicatorLayout createIndicatorLayout(Class<? extends IndicatorLayout> clazz, Context context, PullToRefreshBase.Mode mode) {
return new DefaultIndicatorLayout(context, mode);
}
}
}