import java.util.Map; public class SuperBuilderWithGenericsAndToBuilder { public static class Parent { A field1; Map items; @java.lang.SuppressWarnings("all") public static abstract class ParentBuilder, B extends ParentBuilder> { @java.lang.SuppressWarnings("all") private A field1; @java.lang.SuppressWarnings("all") private java.util.ArrayList items$key; @java.lang.SuppressWarnings("all") private java.util.ArrayList items$value; @java.lang.SuppressWarnings("all") protected B $fillValuesFrom(final C instance) { ParentBuilder.$fillValuesFromInstanceIntoBuilder(instance, this); return self(); } @java.lang.SuppressWarnings("all") private static void $fillValuesFromInstanceIntoBuilder(final Parent instance, final ParentBuilder b) { b.field1(instance.field1); b.items(instance.items == null ? java.util.Collections.emptyMap() : instance.items); } @java.lang.SuppressWarnings("all") protected abstract B self(); @java.lang.SuppressWarnings("all") public abstract C build(); @java.lang.SuppressWarnings("all") public B field1(final A field1) { this.field1 = field1; return self(); } @java.lang.SuppressWarnings("all") public B item(final Integer itemKey, final String itemValue) { if (this.items$key == null) { this.items$key = new java.util.ArrayList(); this.items$value = new java.util.ArrayList(); } this.items$key.add(itemKey); this.items$value.add(itemValue); return self(); } @java.lang.SuppressWarnings("all") public B items(final java.util.Map items) { if (this.items$key == null) { this.items$key = new java.util.ArrayList(); this.items$value = new java.util.ArrayList(); } for (final java.util.Map.Entry $lombokEntry : items.entrySet()) { this.items$key.add($lombokEntry.getKey()); this.items$value.add($lombokEntry.getValue()); } return self(); } @java.lang.SuppressWarnings("all") public B clearItems() { if (this.items$key != null) { this.items$key.clear(); this.items$value.clear(); } return self(); } @java.lang.Override @java.lang.SuppressWarnings("all") public java.lang.String toString() { return "SuperBuilderWithGenericsAndToBuilder.Parent.ParentBuilder(field1=" + this.field1 + ", items$key=" + this.items$key + ", items$value=" + this.items$value + ")"; } } @java.lang.SuppressWarnings("all") private static final class ParentBuilderImpl extends ParentBuilder, ParentBuilderImpl> { @java.lang.SuppressWarnings("all") private ParentBuilderImpl() { } @java.lang.Override @java.lang.SuppressWarnings("all") protected ParentBuilderImpl self() { return this; } @java.lang.Override @java.lang.SuppressWarnings("all") public Parent build() { return new Parent(this); } } @java.lang.SuppressWarnings("all") protected Parent(final ParentBuilder b) { this.field1 = b.field1; java.util.Map items; switch (b.items$key == null ? 0 : b.items$key.size()) { case 0: items = java.util.Collections.emptyMap(); break; case 1: items = java.util.Collections.singletonMap(b.items$key.get(0), b.items$value.get(0)); break; default: items = new java.util.LinkedHashMap(b.items$key.size() < 1073741824 ? 1 + b.items$key.size() + (b.items$key.size() - 3) / 3 : java.lang.Integer.MAX_VALUE); for (int $i = 0; $i < b.items$key.size(); $i++) items.put(b.items$key.get($i), (String) b.items$value.get($i)); items = java.util.Collections.unmodifiableMap(items); } this.items = items; } @java.lang.SuppressWarnings("all") public static ParentBuilder builder() { return new ParentBuilderImpl(); } @java.lang.SuppressWarnings("all") public ParentBuilder toBuilder() { return new ParentBuilderImpl().$fillValuesFrom(this); } } public static class Child extends Parent { double field3; @java.lang.SuppressWarnings("all") public static abstract class ChildBuilder, B extends ChildBuilder> extends Parent.ParentBuilder { @java.lang.SuppressWarnings("all") private double field3; @java.lang.Override @java.lang.SuppressWarnings("all") protected B $fillValuesFrom(final C instance) { super.$fillValuesFrom(instance); ChildBuilder.$fillValuesFromInstanceIntoBuilder(instance, this); return self(); } @java.lang.SuppressWarnings("all") private static void $fillValuesFromInstanceIntoBuilder(final Child instance, final ChildBuilder b) { b.field3(instance.field3); } @java.lang.Override @java.lang.SuppressWarnings("all") protected abstract B self(); @java.lang.Override @java.lang.SuppressWarnings("all") public abstract C build(); @java.lang.SuppressWarnings("all") public B field3(final double field3) { this.field3 = field3; return self(); } @java.lang.Override @java.lang.SuppressWarnings("all") public java.lang.String toString() { return "SuperBuilderWithGenericsAndToBuilder.Child.ChildBuilder(super=" + super.toString() + ", field3=" + this.field3 + ")"; } } @java.lang.SuppressWarnings("all") private static final class ChildBuilderImpl extends ChildBuilder, ChildBuilderImpl> { @java.lang.SuppressWarnings("all") private ChildBuilderImpl() { } @java.lang.Override @java.lang.SuppressWarnings("all") protected ChildBuilderImpl self() { return this; } @java.lang.Override @java.lang.SuppressWarnings("all") public Child build() { return new Child(this); } } @java.lang.SuppressWarnings("all") protected Child(final ChildBuilder b) { super(b); this.field3 = b.field3; } @java.lang.SuppressWarnings("all") public static ChildBuilder builder() { return new ChildBuilderImpl(); } @java.lang.SuppressWarnings("all") public ChildBuilder toBuilder() { return new ChildBuilderImpl().$fillValuesFrom(this); } } public static void test() { Child x = Child.builder().field3(0.0).field1(5).item(5, "").build().toBuilder().build(); } }