ShapeShift
  • Overview
  • Introduction
    • Installation
    • Quick Start
  • API Documentation
    • Annotations
    • Kotlin DSL
    • Java Builder
  • Features
    • Transformers
    • Auto Mapping
    • Conditions
    • Decorators
    • Object Suppliers
    • Mapping Strategy
  • Guides
    • Implicit Transformers
    • Instance Mapping
    • Advanced Mapping
    • Spring Usage
    • Android Usage
Powered by GitBook
On this page
  • Adding Object Suppliers
  • Class Object Suppliers
  • Inline Object Suppliers
  • Mapping with Object Suppliers

Was this helpful?

Export as PDF
  1. Features

Object Suppliers

Mapping to destination classes without a no arg constructor.

PreviousDecoratorsNextMapping Strategy

Last updated 2 years ago

Was this helpful?

Due to the fact that ShapeShift uses reflection behind the scenes, destination classes need a no arg constructor. But in some cases you have no control over the destination classes and cannot modify them to add a no arg constructor. This is where Object Suppliers comes into play, you can register object suppliers to the ShapeShift instance to add your own logic for instance generation.

is another solution for the no arg constructor issue. The advantage of Object Suppliers is that they also work when mapping classes containing subclasses while Instance Mapping only solves the issue for the top level class instance.

We have two classes, the source class SimpleEntity and the destination class SimpleEntityDisplay.

@DefaultMappingTarget(SimpleEntityDisplay::class)
data class SimpleEntity(
    @MappedField
    val name: String,
    @MappedField
    val description: String,
    val privateData: String
)
@DefaultMappingTarget(SimpleEntityDisplay.class)
public class SimpleEntity {
    @MappedField
    private String name;
    @MappedField
    private String description;
    private String privateData;
    // Getters and Setters...
}

The destination class does not have a no arg constructor.

data class SimpleEntityDisplay(
    val name: String,
    val description: String
)
public class SimpleEntityDisplay {
    private String name;
    private String description;

    public SimpleEntityDisplay(String name, String description) {
        this.name = name;
        this.description = description;
    }
    // Getters and Setters...
}

To solve this issue we need to either use or add an object supplier for the class.

Adding Object Suppliers

Adding object suppliers is available through the ShapeShiftBuilder class. Object suppliers can be added inline or as a separate class.

Class Object Suppliers

To create an object supplier class implement the Supplier interface.

class SimpleEntityDisplaySupplier : Supplier<SimpleEntityDisplay> {
    override fun get(): SimpleEntityDisplay {
        return SimpleEntityDisplay("", "")
    }
}
public class SimpleEntityDisplaySupplier implements Supplier<SimpleEntityDisplay> {
    @Override
    public SimpleEntityDisplay get() {
        return new SimpleEntityDisplay("", "");
    }
}

And register it to the ShapeShift instance.

val shapeShift = ShapeShiftBuilder()
    .withObjectSupplier(SimpleEntityDisplaySupplier())
    .build()
ShapeShift shapeShift = new ShapeShiftBuilder()
        .withObjectSupplier(new SimpleEntityDisplaySupplier(), SimpleEntityDisplay.class)
        .build();

Inline Object Suppliers

It is also possible to add the object supplier logic inline.

val shapeShift = ShapeShiftBuilder()
    .withObjectSupplier { SimpleEntityDisplay("", "") }
    .build()
ShapeShift shapeShift = new ShapeShiftBuilder()
        .withObjectSupplier(() -> new SimpleEntityDisplay("", ""), SimpleEntityDisplay.class)
        .build();

Mapping with Object Suppliers

Now that we added an object supplier for the SimpleEntityDisplay class we can map to it as if it has a no arg constructor.

val simpleEntity = SimpleEntity("test", "test description", "private data")
val simpleEntityDisplay = SimpleEntityDisplay("", "")
val result = shapeShift.map<SimpleEntityDisplay>(simpleEntity)
SimpleEntity simpleEntity = new SimpleEntity();
simpleEntity.setName("test");
simpleEntity.setDescription("test description");
simpleEntity.setPrivateData("private data");
SimpleEntityDisplay result = shapeShift.map(simpleEntity, SimpleEntityDisplay.class);
Instance Mapping
instance mapping