CCM NG: Cleanup of PageModel, first part of theming
git-svn-id: https://svn.libreccm.org/ccm/ccm_ng@5037 8810af33-2d31-482b-a856-94f89814c4dfccm-docs
parent
99b3e513aa
commit
cdc2904a3b
|
|
@ -18,7 +18,10 @@
|
||||||
*/
|
*/
|
||||||
package org.libreccm.pagemodel;
|
package org.libreccm.pagemodel;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
import javax.inject.Inject;
|
import javax.inject.Inject;
|
||||||
|
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -32,7 +35,7 @@ import java.util.Optional;
|
||||||
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public abstract class AbstractPageBuilder<P> implements PageBuilder<P> {
|
public abstract class AbstractPageBuilder implements PageBuilder {
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
private ComponentBuilderManager componentBuilderManager;
|
private ComponentBuilderManager componentBuilderManager;
|
||||||
|
|
@ -49,14 +52,16 @@ public abstract class AbstractPageBuilder<P> implements PageBuilder<P> {
|
||||||
* @return A page containing all components from the {@link PageModel}.
|
* @return A page containing all components from the {@link PageModel}.
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public P buildPage(final PageModel pageModel) {
|
public Map<String, Object> buildPage(final PageModel pageModel) {
|
||||||
final P page = buildPage();
|
|
||||||
|
final Map<String, Object> page = buildPage();
|
||||||
|
|
||||||
for (final ComponentModel componentModel : pageModel.getComponents()) {
|
for (final ComponentModel componentModel : pageModel.getComponents()) {
|
||||||
final Optional<Object> component = buildComponent(
|
final Optional<Object> component = buildComponent(
|
||||||
componentModel, componentModel.getClass());
|
componentModel, componentModel.getClass());
|
||||||
if (component.isPresent()) {
|
if (component.isPresent()) {
|
||||||
addComponent(page, component);
|
page.put(componentModel.getIdAttribute(),
|
||||||
|
component);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -77,35 +82,18 @@ public abstract class AbstractPageBuilder<P> implements PageBuilder<P> {
|
||||||
final ComponentModel componentModel,
|
final ComponentModel componentModel,
|
||||||
final Class<M> componentModelClass) {
|
final Class<M> componentModelClass) {
|
||||||
|
|
||||||
componentBuilderManager.findComponentBuilder(componentModel.getClass(),
|
componentBuilderManager.findComponentBuilder(componentModel.getClass());
|
||||||
getType());
|
|
||||||
|
|
||||||
final Optional<ComponentBuilder<M, ?>> builder = componentBuilderManager
|
final Optional<ComponentBuilder<M>> builder = componentBuilderManager
|
||||||
.findComponentBuilder(componentModelClass, getType());
|
.findComponentBuilder(componentModelClass);
|
||||||
|
|
||||||
if (builder.isPresent()) {
|
if (builder.isPresent()) {
|
||||||
return Optional.of(builder.get().buildComponent((M) componentModel));
|
@SuppressWarnings("unchecked")
|
||||||
|
final M model = (M) componentModel;
|
||||||
|
return Optional.of(builder.get().buildComponent(model));
|
||||||
} else {
|
} else {
|
||||||
return Optional.empty();
|
return Optional.empty();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Abstract method returning the type (view technology) for which the
|
|
||||||
* {@link PageBuilder} processes {@link PageModel}s.
|
|
||||||
*
|
|
||||||
* @return
|
|
||||||
*/
|
|
||||||
protected abstract String getType();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A helper method for adding components to the page. How this is done
|
|
||||||
* depends on the view technology, therefore this method must be implemented
|
|
||||||
* by the implementations of this abstract class.
|
|
||||||
*
|
|
||||||
* @param page The page to which the component is added.
|
|
||||||
* @param component The component to add to the page.
|
|
||||||
*/
|
|
||||||
protected abstract void addComponent(P page, Object component);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -18,6 +18,8 @@
|
||||||
*/
|
*/
|
||||||
package org.libreccm.pagemodel;
|
package org.libreccm.pagemodel;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A {@code ComponentBuilder} transforms a {@link ComponentModel} into a
|
* A {@code ComponentBuilder} transforms a {@link ComponentModel} into a
|
||||||
* component.
|
* component.
|
||||||
|
|
@ -25,10 +27,9 @@ package org.libreccm.pagemodel;
|
||||||
*
|
*
|
||||||
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
* @param <M> Type of the model the component builder processes.
|
* @param <M> Type of the model the component builder processes.
|
||||||
* @param <C> Type of the component which is build from the model.
|
|
||||||
*/
|
*/
|
||||||
public interface ComponentBuilder<M extends ComponentModel, C> {
|
public interface ComponentBuilder<M extends ComponentModel> {
|
||||||
|
|
||||||
C buildComponent(M componentModel);
|
Map<String, Object> buildComponent(M componentModel);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -41,11 +41,11 @@ public class ComponentBuilderManager {
|
||||||
ComponentBuilderManager.class);
|
ComponentBuilderManager.class);
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
private Instance<ComponentBuilder<?, ?>> componentBuilders;
|
private Instance<ComponentBuilder<?>> componentBuilders;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Find an implementation of the {@link ComponentBuilder} interface for a
|
* Find an implementation of the {@link ComponentBuilder} interface for a
|
||||||
* specific {@link ComponentModel} and type.
|
* specific {@link ComponentModel}.
|
||||||
*
|
*
|
||||||
* @param <M> Generic variable for the subtype of
|
* @param <M> Generic variable for the subtype of
|
||||||
* {@link ComponentModel} which is produced by
|
* {@link ComponentModel} which is produced by
|
||||||
|
|
@ -53,29 +53,24 @@ public class ComponentBuilderManager {
|
||||||
* @param componentModelClass The sub class of the {@link ComponentModel}
|
* @param componentModelClass The sub class of the {@link ComponentModel}
|
||||||
* for which is processed by the
|
* for which is processed by the
|
||||||
* {@link ComponentBuilder}.
|
* {@link ComponentBuilder}.
|
||||||
* @param type The type for which the
|
|
||||||
* {@link ComponentBuilder} produces the
|
|
||||||
* component(s).
|
|
||||||
*
|
*
|
||||||
* @return An {@link Optional} containing the implementation of the
|
* @return An {@link Optional} containing the implementation of the
|
||||||
* {@link ComponentBuilder} interface for the specified parameters.
|
* {@link ComponentBuilder} interface for the specified parameters.
|
||||||
* If there is no implementation of the specified parameters an
|
* If there is no implementation for the specified parameters an
|
||||||
* empty {@link Optional} is returned.
|
* empty {@link Optional} is returned.
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
public <M extends ComponentModel> Optional<ComponentBuilder<M, ?>> findComponentBuilder(
|
public <M extends ComponentModel> Optional<ComponentBuilder<M>> findComponentBuilder(
|
||||||
final Class<M> componentModelClass,
|
final Class<M> componentModelClass) {
|
||||||
final String type) {
|
|
||||||
|
|
||||||
LOGGER.debug("Trying to find ComponentBuilder for ComponentModel\"{}\""
|
LOGGER.debug("Trying to find ComponentBuilder for ComponentModel\"{}\""
|
||||||
+ "and type \"{}\"...",
|
+ "and type \"{}\"...",
|
||||||
componentModelClass.getName(),
|
componentModelClass.getName());
|
||||||
type);
|
|
||||||
|
|
||||||
final ComponentModelTypeLiteral literal = new ComponentModelTypeLiteral(
|
final ComponentModelTypeLiteral literal = new ComponentModelTypeLiteral(
|
||||||
componentModelClass, type);
|
componentModelClass);
|
||||||
|
|
||||||
final Instance<ComponentBuilder<?, ?>> instance = componentBuilders
|
final Instance<ComponentBuilder<?>> instance = componentBuilders
|
||||||
.select(literal);
|
.select(literal);
|
||||||
if (instance.isUnsatisfied()) {
|
if (instance.isUnsatisfied()) {
|
||||||
LOGGER.warn("No ComponentBuilder for component model \"%s\" "
|
LOGGER.warn("No ComponentBuilder for component model \"%s\" "
|
||||||
|
|
@ -83,16 +78,15 @@ public class ComponentBuilderManager {
|
||||||
return Optional.empty();
|
return Optional.empty();
|
||||||
} else if (instance.isAmbiguous()) {
|
} else if (instance.isAmbiguous()) {
|
||||||
throw new IllegalStateException(String.format(
|
throw new IllegalStateException(String.format(
|
||||||
"Multiple ComponentBuilders for component model \"%s\" and "
|
"Multiple ComponentBuilders for component model \"%s\"available. "
|
||||||
+ "type \"%s\" available. Something is wrong",
|
+ "Something is wrong",
|
||||||
componentModelClass.getName(),
|
componentModelClass.getName()));
|
||||||
type));
|
|
||||||
} else {
|
} else {
|
||||||
final Iterator<ComponentBuilder<?, ?>> iterator = instance.
|
final Iterator<ComponentBuilder<?>> iterator = instance.
|
||||||
iterator();
|
iterator();
|
||||||
final ComponentBuilder<?, ?> componentBuilder = iterator.next();
|
final ComponentBuilder<?> componentBuilder = iterator.next();
|
||||||
|
|
||||||
return Optional.of((ComponentBuilder<M, ?>) componentBuilder);
|
return Optional.of((ComponentBuilder<M>) componentBuilder);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
@ -104,13 +98,10 @@ public class ComponentBuilderManager {
|
||||||
private static final long serialVersionUID = -2601632434295178600L;
|
private static final long serialVersionUID = -2601632434295178600L;
|
||||||
|
|
||||||
private final Class<? extends ComponentModel> componentModel;
|
private final Class<? extends ComponentModel> componentModel;
|
||||||
private final String type;
|
|
||||||
|
|
||||||
public ComponentModelTypeLiteral(
|
public ComponentModelTypeLiteral(
|
||||||
final Class<? extends ComponentModel> componentModel,
|
final Class<? extends ComponentModel> componentModel) {
|
||||||
final String type) {
|
|
||||||
this.componentModel = componentModel;
|
this.componentModel = componentModel;
|
||||||
this.type = type;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
@ -118,11 +109,6 @@ public class ComponentBuilderManager {
|
||||||
return componentModel;
|
return componentModel;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public String type() {
|
|
||||||
return type;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,5 @@ public @interface ComponentModelType {
|
||||||
|
|
||||||
Class<? extends ComponentModel> componentModel();
|
Class<? extends ComponentModel> componentModel();
|
||||||
|
|
||||||
String type();
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -18,6 +18,8 @@
|
||||||
*/
|
*/
|
||||||
package org.libreccm.pagemodel;
|
package org.libreccm.pagemodel;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
import javax.enterprise.context.RequestScoped;
|
import javax.enterprise.context.RequestScoped;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -32,9 +34,8 @@ import javax.enterprise.context.RequestScoped;
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
* @param <P> The type of page the page builder creates.
|
|
||||||
*/
|
*/
|
||||||
public interface PageBuilder<P> {
|
public interface PageBuilder {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Build a page for the view technology supported by this page builder
|
* Build a page for the view technology supported by this page builder
|
||||||
|
|
@ -44,7 +45,7 @@ public interface PageBuilder<P> {
|
||||||
*
|
*
|
||||||
* @return A page with the default components.
|
* @return A page with the default components.
|
||||||
*/
|
*/
|
||||||
P buildPage();
|
Map<String, Object> buildPage();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Build a page of type {@code P} using the provided {@link PageModel}.
|
* Build a page of type {@code P} using the provided {@link PageModel}.
|
||||||
|
|
@ -55,7 +56,7 @@ public interface PageBuilder<P> {
|
||||||
*
|
*
|
||||||
* @return The page generated from the provided {@link PageModel}.
|
* @return The page generated from the provided {@link PageModel}.
|
||||||
*/
|
*/
|
||||||
P buildPage(PageModel pageModel);
|
Map<String, Object> buildPage(PageModel pageModel);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,124 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (C) 2016 LibreCCM Foundation.
|
|
||||||
*
|
|
||||||
* This library is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU Lesser General Public
|
|
||||||
* License as published by the Free Software Foundation; either
|
|
||||||
* version 2.1 of the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This library is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this library; if not, write to the Free Software
|
|
||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
||||||
* MA 02110-1301 USA
|
|
||||||
*/
|
|
||||||
package org.libreccm.pagemodel;
|
|
||||||
|
|
||||||
import org.apache.logging.log4j.LogManager;
|
|
||||||
import org.apache.logging.log4j.Logger;
|
|
||||||
import org.libreccm.web.CcmApplication;
|
|
||||||
|
|
||||||
import javax.enterprise.context.RequestScoped;
|
|
||||||
import javax.enterprise.inject.Instance;
|
|
||||||
import javax.enterprise.util.AnnotationLiteral;
|
|
||||||
import javax.inject.Inject;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.Optional;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Provides access to all available {@link PageBuilder} implementations.
|
|
||||||
*
|
|
||||||
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
|
||||||
*/
|
|
||||||
@RequestScoped
|
|
||||||
public class PageBuilderManager {
|
|
||||||
|
|
||||||
private static final Logger LOGGER = LogManager.getLogger(
|
|
||||||
PageBuilderManager.class);
|
|
||||||
|
|
||||||
@Inject
|
|
||||||
private Instance<PageBuilder<?>> pageBuilders;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Find a {@link PageBuilder} for a specific type and application type.
|
|
||||||
*
|
|
||||||
* @param type The type of the {@link PageBuilder}.
|
|
||||||
* @param applicationType The application type for which the
|
|
||||||
* {@link PageBuilder} builds pages.
|
|
||||||
*
|
|
||||||
* @return An {@link Optional} containing the {@link PageBuilder}
|
|
||||||
* implementation for the specified {@code type} and
|
|
||||||
* {@code applicationType}. If there is no {@code PageBuilder} for
|
|
||||||
* the specified parameters an empty {@link Optional} is returned.
|
|
||||||
*/
|
|
||||||
public Optional<PageBuilder<?>> findPageBuilder(
|
|
||||||
final String type,
|
|
||||||
final Class<? extends CcmApplication> applicationType) {
|
|
||||||
|
|
||||||
LOGGER.debug("Trying to find PageBuilder for type \"{}\" and "
|
|
||||||
+ "application type \"{}\"...",
|
|
||||||
type,
|
|
||||||
applicationType);
|
|
||||||
|
|
||||||
final PageModelTypeLiteral literal = new PageModelTypeLiteral(
|
|
||||||
type, applicationType);
|
|
||||||
|
|
||||||
final Instance<PageBuilder<?>> instance = pageBuilders.select(literal);
|
|
||||||
if (instance.isUnsatisfied()) {
|
|
||||||
LOGGER.warn("No PageBuilder for type \"{}\" and application type "
|
|
||||||
+ "\"{}\" available.",
|
|
||||||
type,
|
|
||||||
applicationType);
|
|
||||||
return Optional.empty();
|
|
||||||
} else if (instance.isAmbiguous()) {
|
|
||||||
throw new IllegalArgumentException(String.format(
|
|
||||||
"Multiple PageBuilders for type \"%s\" and "
|
|
||||||
+ "application type \"%s\" avilable. Something is wrong.",
|
|
||||||
type,
|
|
||||||
applicationType));
|
|
||||||
} else {
|
|
||||||
LOGGER.debug("Found PageBuilder for type \"{}\" and application "
|
|
||||||
+ "type \"{}\"...",
|
|
||||||
type,
|
|
||||||
applicationType);
|
|
||||||
final Iterator<PageBuilder<?>> iterator = instance.iterator();
|
|
||||||
final PageBuilder<?> pageBuilder = iterator.next();
|
|
||||||
|
|
||||||
return Optional.of(pageBuilder);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private class PageModelTypeLiteral
|
|
||||||
extends AnnotationLiteral<PageModelType>
|
|
||||||
implements PageModelType {
|
|
||||||
|
|
||||||
private static final long serialVersionUID = 5919950993273871601L;
|
|
||||||
|
|
||||||
private final String type;
|
|
||||||
private final Class<? extends CcmApplication> applicationType;
|
|
||||||
|
|
||||||
public PageModelTypeLiteral(
|
|
||||||
final String type,
|
|
||||||
final Class<? extends CcmApplication> applicationType) {
|
|
||||||
|
|
||||||
this.type = type;
|
|
||||||
this.applicationType = applicationType;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String type() {
|
|
||||||
return type;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Class<? extends CcmApplication> applicationType() {
|
|
||||||
return applicationType;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
@ -1,70 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (C) 2016 LibreCCM Foundation.
|
|
||||||
*
|
|
||||||
* This library is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU Lesser General Public
|
|
||||||
* License as published by the Free Software Foundation; either
|
|
||||||
* version 2.1 of the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This library is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this library; if not, write to the Free Software
|
|
||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
||||||
* MA 02110-1301 USA
|
|
||||||
*/
|
|
||||||
package org.libreccm.pagemodel.bebop;
|
|
||||||
|
|
||||||
import com.arsdigita.bebop.Component;
|
|
||||||
import com.arsdigita.bebop.Page;
|
|
||||||
import com.arsdigita.bebop.PageFactory;
|
|
||||||
import com.arsdigita.web.Web;
|
|
||||||
import org.libreccm.pagemodel.AbstractPageBuilder;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Basic implementation of a {@link PageBuilder} for Bebop {@link Page}s.
|
|
||||||
* Applications must provided an implementation of the {@link PageBuilder}.
|
|
||||||
* These implementations must override the
|
|
||||||
* {@link #addDefaultComponents(com.arsdigita.bebop.Page)} method.
|
|
||||||
*
|
|
||||||
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
|
||||||
*/
|
|
||||||
public abstract class AbstractBebopPageBuilder extends AbstractPageBuilder<Page> {
|
|
||||||
|
|
||||||
public static final String BEBOP = "Bebop";
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected String getType() {
|
|
||||||
return BEBOP;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void addComponent(final Page page, final Object component) {
|
|
||||||
final Component bebopComponent = (Component) component;
|
|
||||||
|
|
||||||
page.add(bebopComponent);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Page buildPage() {
|
|
||||||
|
|
||||||
final String application = Web.getWebContext().getApplication().
|
|
||||||
getPrimaryUrl();
|
|
||||||
final Page page = PageFactory.buildPage(application, "");
|
|
||||||
|
|
||||||
addDefaultComponents(page);
|
|
||||||
|
|
||||||
return page;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add the default components which are present on every page.
|
|
||||||
*
|
|
||||||
* @param page The page to which the components are added.
|
|
||||||
*/
|
|
||||||
public abstract void addDefaultComponents(final Page page);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
@ -0,0 +1,144 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2017 LibreCCM Foundation.
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
||||||
|
* MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
package org.libreccm.theming;
|
||||||
|
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Informations about a file in a theme.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
|
*/
|
||||||
|
public class ThemeFileInfo {
|
||||||
|
|
||||||
|
private String name;
|
||||||
|
private boolean directory;
|
||||||
|
private String mimeType;
|
||||||
|
private long size;
|
||||||
|
private boolean writable;
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(final String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isDirectory() {
|
||||||
|
return directory;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDirectory(final boolean directory) {
|
||||||
|
this.directory = directory;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getMimeType() {
|
||||||
|
return mimeType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMimeType(final String mimeType) {
|
||||||
|
this.mimeType = mimeType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getSize() {
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSize(final long size) {
|
||||||
|
this.size = size;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isWritable() {
|
||||||
|
return writable;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setWritable(final boolean writable) {
|
||||||
|
this.writable = writable;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
int hash = 3;
|
||||||
|
hash = 83 * hash + Objects.hashCode(name);
|
||||||
|
hash = 83 * hash + (directory ? 1 : 0);
|
||||||
|
hash = 83 * hash + Objects.hashCode(mimeType);
|
||||||
|
hash = 83 * hash + (int) (size ^ (size >>> 32));
|
||||||
|
hash = 83 * hash + (writable ? 1 : 0);
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(final Object obj) {
|
||||||
|
if (this == obj) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (obj == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!(obj instanceof ThemeFileInfo)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
final ThemeFileInfo other = (ThemeFileInfo) obj;
|
||||||
|
if (!other.canEqual(this)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (directory != other.isDirectory()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (size != other.getSize()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (writable != other.isWritable()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!Objects.equals(name, other.getName())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return Objects.equals(mimeType, other.getMimeType());
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean canEqual(final Object obj) {
|
||||||
|
return obj instanceof ThemeFileInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public final String toString() {
|
||||||
|
return toString("");
|
||||||
|
}
|
||||||
|
|
||||||
|
public String toString(final String data) {
|
||||||
|
return String.format("%s{ "
|
||||||
|
+ "name = \"%s\", "
|
||||||
|
+ "directory = %b, "
|
||||||
|
+ "mimeType = \"%s\", "
|
||||||
|
+ "size = %d, "
|
||||||
|
+ "writable = %b%s"
|
||||||
|
+ " }",
|
||||||
|
super.toString(),
|
||||||
|
name,
|
||||||
|
directory,
|
||||||
|
mimeType,
|
||||||
|
size,
|
||||||
|
writable,
|
||||||
|
data);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,124 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2017 LibreCCM Foundation.
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
||||||
|
* MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
package org.libreccm.theming;
|
||||||
|
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
|
*/
|
||||||
|
public class ThemeInfo {
|
||||||
|
|
||||||
|
private String name;
|
||||||
|
|
||||||
|
private ThemeVersion version;
|
||||||
|
|
||||||
|
private String type;
|
||||||
|
|
||||||
|
private Class<ThemeProvider> provider;
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(final String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ThemeVersion getVersion() {
|
||||||
|
return version;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setVersion(final ThemeVersion version) {
|
||||||
|
this.version = version;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getType() {
|
||||||
|
return type;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setType(final String type) {
|
||||||
|
this.type = type;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Class<ThemeProvider> getProvider() {
|
||||||
|
return provider;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setProvider(final Class<ThemeProvider> provider) {
|
||||||
|
this.provider = provider;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
int hash = 5;
|
||||||
|
hash = 73 * hash + Objects.hashCode(name);
|
||||||
|
hash = 73 * hash + Objects.hashCode(version);
|
||||||
|
hash = 73 * hash + Objects.hashCode(type);
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object obj) {
|
||||||
|
if (this == obj) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (obj == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!(obj instanceof ThemeInfo)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
final ThemeInfo other = (ThemeInfo) obj;
|
||||||
|
if (!other.canEqual(this)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!Objects.equals(name, other.getName())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!Objects.equals(type, other.getType())) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return version == other.getVersion();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean canEqual(final Object obj) {
|
||||||
|
return obj instanceof ThemeInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public final String toString() {
|
||||||
|
return toString("");
|
||||||
|
}
|
||||||
|
|
||||||
|
public String toString(final String data) {
|
||||||
|
return String.format("%s{ "
|
||||||
|
+ "name = \"%s\", "
|
||||||
|
+ "version = %s, "
|
||||||
|
+ "type = \"%s\"%s"
|
||||||
|
+ " }",
|
||||||
|
super.toString(),
|
||||||
|
name,
|
||||||
|
Objects.toString(version),
|
||||||
|
type,
|
||||||
|
data);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (C) 2016 LibreCCM Foundation.
|
* Copyright (C) 2017 LibreCCM Foundation.
|
||||||
*
|
*
|
||||||
* This library is free software; you can redistribute it and/or
|
* This library is free software; you can redistribute it and/or
|
||||||
* modify it under the terms of the GNU Lesser General Public
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
|
@ -16,29 +16,18 @@
|
||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
||||||
* MA 02110-1301 USA
|
* MA 02110-1301 USA
|
||||||
*/
|
*/
|
||||||
package org.libreccm.pagemodel;
|
package org.libreccm.theming;
|
||||||
|
|
||||||
import org.libreccm.web.CcmApplication;
|
import java.util.Map;
|
||||||
|
|
||||||
import javax.inject.Qualifier;
|
|
||||||
import java.lang.annotation.ElementType;
|
|
||||||
import java.lang.annotation.Retention;
|
|
||||||
import java.lang.annotation.RetentionPolicy;
|
|
||||||
import java.lang.annotation.Target;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Specifies for which application type and which view technology a
|
|
||||||
* {@link PageBuilder} builds the pages.
|
|
||||||
*
|
*
|
||||||
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
*/
|
*/
|
||||||
@Qualifier
|
public interface ThemeProcessor {
|
||||||
@Retention(RetentionPolicy.RUNTIME)
|
|
||||||
@Target(ElementType.TYPE)
|
|
||||||
public @interface PageModelType {
|
|
||||||
|
|
||||||
String type();
|
String process(Map<String, Object> page,
|
||||||
|
ThemeInfo theme,
|
||||||
Class<? extends CcmApplication> applicationType();
|
ThemeProvider themeProvider);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
@ -0,0 +1,155 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2017 LibreCCM Foundation.
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
||||||
|
* MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
package org.libreccm.theming;
|
||||||
|
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A theme provider provides access to the files of a theme. It abstracts from
|
||||||
|
* the location and method of loading.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
|
*/
|
||||||
|
public interface ThemeProvider {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides a list of all themes provided by this theme provider. The list
|
||||||
|
* should be ordered by the name of the theme.
|
||||||
|
*
|
||||||
|
* @return A list of all themes provided by this theme provider. If the
|
||||||
|
* implementation supports draft and live themes the list should
|
||||||
|
* contain all draft themes.
|
||||||
|
*/
|
||||||
|
List<ThemeInfo> getThemes();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides a list of all live themes provided by this theme provider.
|
||||||
|
*
|
||||||
|
* @return A list of all live themes provided by this theme provider. If the
|
||||||
|
* implementation does not support draft/live themes the
|
||||||
|
* implementation of this method returns the same list as
|
||||||
|
* {@link #getThemes()}.
|
||||||
|
*/
|
||||||
|
List<ThemeInfo> getLiveThemes();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide information about a theme.
|
||||||
|
*
|
||||||
|
* @param theme The theme.
|
||||||
|
* @param version The version of the theme. Implementations which do not
|
||||||
|
* support draft/live themes will ignore this parameter.
|
||||||
|
*
|
||||||
|
* @return Informations about the theme identified by the provided name. If
|
||||||
|
* there is no such theme provided by this {@code ThemeProvider} an
|
||||||
|
* empty optional is returned.
|
||||||
|
*/
|
||||||
|
Optional<ThemeInfo> getThemeInfo(String theme, ThemeVersion version);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List all files in a theme at the specified path.
|
||||||
|
*
|
||||||
|
* @param theme The theme of which the files are listed.
|
||||||
|
* @param version The version of the theme for which the files are listed.
|
||||||
|
* Implementations which do not support draft/live themes
|
||||||
|
* will ignore this parameter.
|
||||||
|
* @param path The path of the directory of which the files are listed.
|
||||||
|
* The path is relative to the root of the theme.To get the
|
||||||
|
* root directory provided an empty string. Implementations
|
||||||
|
* should throw an NullPointerException if {@code null} is
|
||||||
|
* provided as path.
|
||||||
|
*
|
||||||
|
* @return A list of all files in the provided directory. If there is such
|
||||||
|
* path in the theme the list is empty. If the path is the path of a
|
||||||
|
* file and not a directory the list should have one element, the
|
||||||
|
* data about the file itself.
|
||||||
|
*/
|
||||||
|
List<ThemeFileInfo> listThemeFiles(String theme,
|
||||||
|
ThemeVersion version,
|
||||||
|
String path);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve a file from a theme. We use an {@link InputStream} here because
|
||||||
|
* that is the most universal interface in the Java API which works for all
|
||||||
|
* sorts of resources and is independent from any other API.
|
||||||
|
*
|
||||||
|
* @param theme The theme from which the file is retrieved.
|
||||||
|
* @param version The version of the theme from which the file is retrieved.
|
||||||
|
* Implementations which do not support draft/live themes
|
||||||
|
* will ignore this parameter.
|
||||||
|
* @param path The path of file to retrieve relative to the root of the
|
||||||
|
* theme.
|
||||||
|
*
|
||||||
|
* @return An {@link Optional} containing an {@link InputStream} for the
|
||||||
|
* requested file or an empty optional if the theme has no such
|
||||||
|
* file.
|
||||||
|
*/
|
||||||
|
Optional<InputStream> getThemeFileAsStream(String theme,
|
||||||
|
ThemeVersion version,
|
||||||
|
String path);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an {@link OutputStream} for a theme file. Implementations which
|
||||||
|
* do not support changes to the theme files should throw an
|
||||||
|
* {@link UnsupportedOperationException}. If the file is not writable for
|
||||||
|
* some reason an {@link IllegalArgumentException} should be thrown.
|
||||||
|
*
|
||||||
|
* If an implementation supports draft/live themes the {@link OutputStream}
|
||||||
|
* always changes the file in the draft version of the theme. The live theme
|
||||||
|
* should only be changed by {@link #publishTheme(String).
|
||||||
|
*
|
||||||
|
* If the file does not exist it the file is created.
|
||||||
|
*
|
||||||
|
* If not all directories in the provided path already exist an
|
||||||
|
* implementation should create the missing directories.
|
||||||
|
*
|
||||||
|
* @param theme The theme to which the file belongs.
|
||||||
|
* @param path The path of the file to update
|
||||||
|
*
|
||||||
|
* @return An {@link OutputStream} for the file.
|
||||||
|
*/
|
||||||
|
OutputStream getOutputStreamForThemeFile(String theme, String path);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determines if the implementation supports changes to the files of the
|
||||||
|
* themes.
|
||||||
|
*
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
boolean supportsChanges();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determines if the implementation supports draft/live themes.
|
||||||
|
*
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
boolean supportsDraftThemes();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Publishes all changes done to a draft theme to its live version. For
|
||||||
|
* implementations which do not support draft/live themes the implementation
|
||||||
|
* of this method should be a noop, but not throw an exception.
|
||||||
|
*
|
||||||
|
* @param theme The theme to publish.
|
||||||
|
*/
|
||||||
|
void publishTheme(String theme);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,29 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2017 LibreCCM Foundation.
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
||||||
|
* MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
package org.libreccm.theming;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
|
*/
|
||||||
|
public @interface ThemeType {
|
||||||
|
|
||||||
|
String value();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,30 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2017 LibreCCM Foundation.
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
||||||
|
* MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
package org.libreccm.theming;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
|
*/
|
||||||
|
public enum ThemeVersion {
|
||||||
|
|
||||||
|
DRAFT,
|
||||||
|
LIVE
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,135 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2017 LibreCCM Foundation.
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
||||||
|
* MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
package org.libreccm.theming;
|
||||||
|
|
||||||
|
import org.apache.logging.log4j.LogManager;
|
||||||
|
import org.apache.logging.log4j.Logger;
|
||||||
|
import org.libreccm.core.UnexpectedErrorException;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import javax.enterprise.context.RequestScoped;
|
||||||
|
import javax.enterprise.inject.Instance;
|
||||||
|
import javax.enterprise.util.AnnotationLiteral;
|
||||||
|
import javax.inject.Inject;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
|
||||||
|
*/
|
||||||
|
@RequestScoped
|
||||||
|
public class Themes {
|
||||||
|
|
||||||
|
private static final Logger LOGGER = LogManager.getLogger(Themes.class);
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
private Instance<ThemeProvider> providers;
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
private Instance<ThemeProcessor> processors;
|
||||||
|
|
||||||
|
public List<ThemeInfo> getAvailableThemes() {
|
||||||
|
|
||||||
|
final List<ThemeInfo> themes = new ArrayList<>();
|
||||||
|
for (final ThemeProvider provider : providers) {
|
||||||
|
themes.addAll(provider.getThemes());
|
||||||
|
}
|
||||||
|
|
||||||
|
return themes;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<ThemeInfo> getLiveThemes() {
|
||||||
|
|
||||||
|
final List<ThemeInfo> themes = new ArrayList<>();
|
||||||
|
for (final ThemeProvider provider : providers) {
|
||||||
|
themes.addAll(provider.getLiveThemes());
|
||||||
|
}
|
||||||
|
|
||||||
|
return themes;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String process(Map<String, Object> page, ThemeInfo theme) {
|
||||||
|
|
||||||
|
final ThemeTypeLiteral themeType = new ThemeTypeLiteral(theme.getType());
|
||||||
|
|
||||||
|
final Instance<ThemeProcessor> forType = processors.select(themeType);
|
||||||
|
if (forType.isUnsatisfied()) {
|
||||||
|
LOGGER.error("No ThemeProcessor implementation for type \"{}\" of "
|
||||||
|
+ "theme \"{}\".",
|
||||||
|
theme.getType(),
|
||||||
|
theme.getName());
|
||||||
|
throw new UnexpectedErrorException(String
|
||||||
|
.format("No ThemeProcessor implementation for type \"%s\" of "
|
||||||
|
+ "theme \"%s\".",
|
||||||
|
theme.getType(),
|
||||||
|
theme.getName()));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (forType.isAmbiguous()) {
|
||||||
|
LOGGER.error(
|
||||||
|
"Mutiple ThemeProcessor implementations for type \"{}\" of "
|
||||||
|
+ "theme \"{}\".",
|
||||||
|
theme.getType(),
|
||||||
|
theme.getName());
|
||||||
|
throw new UnexpectedErrorException(String
|
||||||
|
.format(
|
||||||
|
"Mutiple ThemeProcessor implementations for type \"%s\" of "
|
||||||
|
+ "theme \"%s\".",
|
||||||
|
theme.getType(),
|
||||||
|
theme.getName()));
|
||||||
|
}
|
||||||
|
|
||||||
|
final Instance<ThemeProvider> forTheme = providers.select(theme
|
||||||
|
.getProvider());
|
||||||
|
|
||||||
|
if (forTheme.isUnsatisfied()) {
|
||||||
|
LOGGER.error("ThemeProvider \"{}\" not found.",
|
||||||
|
theme.getProvider().getName());
|
||||||
|
throw new UnexpectedErrorException(String.format(
|
||||||
|
"ThemeProvider \"%s\" not found.",
|
||||||
|
theme.getProvider().getName()));
|
||||||
|
}
|
||||||
|
|
||||||
|
final ThemeProcessor processor = forType.get();
|
||||||
|
final ThemeProvider provider = forTheme.get();
|
||||||
|
|
||||||
|
return processor.process(page, theme, provider);
|
||||||
|
}
|
||||||
|
|
||||||
|
private class ThemeTypeLiteral extends AnnotationLiteral<ThemeType>
|
||||||
|
implements ThemeType {
|
||||||
|
|
||||||
|
private static final long serialVersionUID = 3377237291286175824L;
|
||||||
|
|
||||||
|
private final String value;
|
||||||
|
|
||||||
|
public ThemeTypeLiteral(final String value) {
|
||||||
|
this.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String value() {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue