/**
*
* Copyright
* 2009-2015 Jayway Products AB
* 2016-2017 Föreningen Sambruk
*
* Licensed under AGPL, Version 3.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.gnu.org/licenses/agpl.txt
*
* 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 se.streamsource.streamflow.web.domain.structure.conversation;
import org.qi4j.api.common.Optional;
import org.qi4j.api.concern.Concerns;
import org.qi4j.api.entity.EntityBuilder;
import org.qi4j.api.entity.Identity;
import org.qi4j.api.entity.IdentityGenerator;
import org.qi4j.api.entity.association.ManyAssociation;
import org.qi4j.api.injection.scope.Service;
import org.qi4j.api.injection.scope.Structure;
import org.qi4j.api.injection.scope.This;
import org.qi4j.api.mixin.Mixins;
import org.qi4j.api.specification.Specification;
import org.qi4j.api.structure.Module;
import org.qi4j.api.util.Iterables;
import se.streamsource.streamflow.api.workspace.cases.conversation.MessageType;
import se.streamsource.streamflow.infrastructure.event.domain.DomainEvent;
import se.streamsource.streamflow.web.domain.interaction.gtd.Unread;
import se.streamsource.streamflow.web.domain.structure.attachment.Attachment;
import se.streamsource.streamflow.web.domain.structure.attachment.Attachments;
/**
* JAVADOC
*/
@Concerns(UpdateCaseCountMessagesConcern.class)
@Mixins(Messages.Mixin.class)
public interface Messages
{
Message createMessage(String body, MessageType messageType, ConversationParticipant participant);
Message createMessage(String body, MessageType messageType, ConversationParticipant participant, boolean unread );
Message getLastMessage();
void createMessageFromDraft( ConversationParticipant participant );
void createMessageFromDraft( ConversationParticipant participant, MessageType messageType );
boolean hasUnreadMessage();
void markRead();
interface Data
{
ManyAssociation<Message> messages();
Message createdMessage( @Optional DomainEvent event, String id, String body, MessageType messageType, ConversationParticipant participant, boolean unread );
Message createdMessageFromDraft( @Optional DomainEvent event, String id, MessageDraft draft, ConversationParticipant participant, MessageType messageType );
}
abstract class Mixin
implements Messages, Data
{
@Service
IdentityGenerator idGen;
@Structure
Module module;
@This
ConversationParticipants participants;
@This
Conversation conversation;
public Message createMessage( String body, MessageType messageType, ConversationParticipant participant ) throws IllegalArgumentException
{
return createMessage( body, messageType, participant, true );
}
public Message createMessage( String body, MessageType messageType, ConversationParticipant participant, boolean unread ) throws IllegalArgumentException
{
if (!participants.isParticipant(participant))
{
participants.addParticipant( participant );
}
Message message = createdMessage( null, idGen.generate( Identity.class ), body, messageType, participant, unread );
participants.receiveMessage(message);
return message;
}
public Message createdMessage( DomainEvent event, String id, String body, MessageType messageType, ConversationParticipant participant, boolean unread )
{
EntityBuilder<Message> builder = module.unitOfWorkFactory().currentUnitOfWork().newEntityBuilder( Message.class, id );
builder.instanceFor( Message.Data.class ).body().set( body );
builder.instanceFor( Message.Data.class ).createdOn().set( event.on().get() );
builder.instanceFor( Message.Data.class ).sender().set( participant );
builder.instanceFor( Message.Data.class ).conversation().set( conversation );
builder.instanceFor( Message.Data.class ).messageType().set( messageType );
if( unread )
{
builder.instanceFor( Unread.Data.class ).unread().set( true );
}
Message message = builder.newInstance();
messages().add( message );
return message;
}
public Message getLastMessage()
{
if (messages().count() > 0)
return messages().get(messages().count()-1);
else
return null;
}
public void createMessageFromDraft( ConversationParticipant participant )
{
createMessageFromDraft( participant, MessageType.PLAIN );
}
public void createMessageFromDraft( ConversationParticipant participant, MessageType messageType )
{
if (!participants.isParticipant(participant))
{
participants.addParticipant( participant );
}
Message message = createdMessageFromDraft( null, idGen.generate( Identity.class ), ((MessageDraft)conversation), participant, messageType );
for( Attachment attachment : ((Attachments.Data)conversation).attachments().toList() )
{
message.addAttachment( attachment );
// remove attachment from draft attachments data so AttachmentEntity does not get
// removed for real - we just moved it to message attachments where it actually belongs after
// message creation.
((Attachments.Data)conversation).attachments().remove( attachment );
}
// also reset draft message body
((MessageDraft)conversation).changeDraftMessage( null );
participants.receiveMessage(message);
}
public Message createdMessageFromDraft( @Optional DomainEvent event, String id, MessageDraft draft, ConversationParticipant participant, MessageType messageType )
{
EntityBuilder<Message> builder = module.unitOfWorkFactory().currentUnitOfWork().newEntityBuilder( Message.class, id );
builder.instanceFor( Message.Data.class ).body().set( ((MessageDraft.Data)draft).draftmessage().get() );
builder.instanceFor( Message.Data.class ).createdOn().set( event.on().get() );
builder.instanceFor( Message.Data.class ).sender().set( participant );
builder.instanceFor( Message.Data.class ).conversation().set( conversation );
builder.instanceFor( Message.Data.class ).messageType().set( messageType );
Message message = builder.newInstance();
messages().add( message );
return message;
}
public boolean hasUnreadMessage()
{
return Iterables.matchesAny( new Specification<Message>()
{
public boolean satisfiedBy( Message msg )
{
return ((Unread.Data) msg).unread().get();
}
}, messages() );
}
public void markRead()
{
for( Message message : messages() )
{
message.setUnread( false );
}
}
}
}