package jpa20mitbeanvalidation.entities;

import java.io.Serializable;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import javax.persistence.Basic;
import javax.persistence.CascadeType;
import javax.persistence.Entity;
import javax.persistence.EntityManager;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.ManyToMany;
import javax.persistence.NamedQueries;
import javax.persistence.NamedQuery;
import javax.persistence.OneToMany;
import javax.persistence.Version;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Size;
import jpa20mitbeanvalidation.validatoren.MrXProgrammer;



@NamedQueries({
    @NamedQuery(name="Mitarbeiter.alle",
                query="SELECT m FROM Mitarbeiter m"),
    @NamedQuery(name="Mitarbeiter.primaryKey",
                query="SELECT m FROM Mitarbeiter m WHERE m.minr = :minr"),
    @NamedQuery(name="Mitarbeiter.mitFaehigkeit",
             query="SELECT m FROM Rolle r JOIN r.mitarbeiter m WHERE r.name= :name"
                )
                                                                          
                                                                          
})
@Entity @MrXProgrammer(sprachen={"Java","Groovy"}, message="aktuelle Sprachen")
public class Mitarbeiter implements Serializable {

  private static final long serialVersionUID = 1L;

  @Id
  @GeneratedValue(strategy = GenerationType.AUTO)
  private int minr;
  @NotNull @Size(min=2, message="echter Nachname")
  private String name;

  @OneToMany(mappedBy = "bearbeiter",
             cascade = {CascadeType.PERSIST, CascadeType.MERGE})
  @Basic(fetch = FetchType.EAGER)
  @Size(max=3, message="max 3 Auftraege")
  private Set<Projektauftrag> auftraege = new HashSet<Projektauftrag>();

  @ManyToMany(mappedBy = "mitarbeiter",
              cascade = {CascadeType.PERSIST, CascadeType.MERGE})
  @Basic(fetch = FetchType.EAGER)
  @Size(max=2, message="max 2 Rollen")
  private List<Rolle> rollen = new ArrayList<Rolle>();

  @Version
  private int version;

  public static Mitarbeiter zuMinr(int minr, EntityManager em){
    return (Mitarbeiter) em.createNamedQuery("Mitarbeiter.primaryKey",Mitarbeiter.class).
            setParameter("minr", minr).getSingleResult();
  }

  public static List<Mitarbeiter> mitFaehigkeit(String name, EntityManager em){
    return (List<Mitarbeiter>) em.createNamedQuery("Mitarbeiter.mitFaehigkeit").
            setParameter("name", name).getResultList();
  }

  public void auftragHinzu(Projektauftrag pa) {
    Set<Projektauftrag> tmp = getAuftraege();
    tmp.add(pa);
    setAuftraege(tmp);
  }

  public void rolleHinzu(Rolle r) {
    List<Rolle> tmp = getRollen();
    tmp.add(r);
    setRollen(tmp);
  }

  public Mitarbeiter() {
  }

  public Mitarbeiter(String name) {
    this.name = name;
  }

  public int getVersion() {
    return version;
  }

  public void setVersion(int version) {
    this.version = version;
  }

  public List<Rolle> getRollen() {
    return rollen;
  }

  public void setRollen(List<Rolle> rollen) {
    this.rollen = rollen;
  }

  public Set<Projektauftrag> getAuftraege() {
    return auftraege;
  }

  public void setAuftraege(Set<Projektauftrag> auftraege) {
    this.auftraege = auftraege;
  }

  public int getMinr() {
    return minr;
  }

  public void setMinr(int id) {
    this.minr = id;
  }

  public String getName() {
    return name;
  }

  public void setName(String name) {
    this.name = name;
  }

  @Override
  public int hashCode() {
    return minr;
  }

  @Override
  public boolean equals(Object object) {
    if (!(object instanceof Mitarbeiter)) {
      return false;
    }
    Mitarbeiter other = (Mitarbeiter) object;
    if (this.minr != other.minr) {
      return false;
    }
    return true;
  }

  @Override
  public String toString() {
    StringBuffer ergebnis = new StringBuffer(minr + ": " + name + " Auftraege=[ ");
    for (Projektauftrag a : auftraege) {
      ergebnis.append(a.getTitel() + " ");
    }
    ergebnis.append("]\n   Rollen=[ ");
    for (Rolle r : rollen) {
      ergebnis.append(r.getName() + " ");
    }
    ergebnis.append("]");
    return ergebnis.toString();
  }
}
