Skip to content

Latest commit

 

History

History
201 lines (145 loc) · 3.53 KB

bs.deriving.md

File metadata and controls

201 lines (145 loc) · 3.53 KB

[@bs.deriving]

[@bs.deriving accessors]

Reason records are represented as arrays in JS.

For example:

type person = {
  name: string,
  age: int,
  country: string,
};

let misha: person = {
  name: "Misha",
  age: 18,
  country: "Australia",
};

compiles to:

var misha = ["Misha", 18, "Australia"];

Imagine that you converted some JS code to Reason and it uses the record above. The code that isn't converted to Reason yet wants to access misha's fields.

In order to get the country, for example, you would need to know to access misha[2].

You shouldn't need to worry about the array representation of a record or the indices where certain fields live.

That's exactly what bs.deriving accessors is for! It creates getter functions for all fields of the annotated record.

For example:

[@bs.deriving accessors]
type person = {
  name: string,
  age: int,
  country: string,
};

let misha: person = {
  name: "Misha",
  age: 18,
  country: "Australia",
};

compiles to:

function name(param) {
  return param[0];
}

function age(param) {
  return param[1];
}

function country(param) {
  return param[2];
}

var misha = ["Misha", 18, "Australia"];

Now, to get misha's country we can use the generated accessor: country(misha)

Similarly, imagine your new Reason code has the following variant:

type action =
  | Add(string)
  | Toggle(int)
  | DeleteOne(int)
  | DeleteAll;

and you want to create DeleteOne(12) from JS, for example.

By adding bs.deriving accessors, you get a creation function for every variant constructor.

[@bs.deriving accessors]
type action =
  | Add(string)
  | Toggle(int)
  | DeleteOne(int)
  | DeleteAll;

Now, in JS you can do:

let actions = [
  add("Drink coffee"), 
  toggle(34),
  deleteOne(12),
  deleteAll
];

[@bs.deriving abstract]

Reason records are represented as arrays in JS:

type person = {
  name: string,
  age: int
};

let misha: person = {
  name: "Misha",
  age: 18
};

Js.log(misha); /* => ["Misha",18] */

To represent them as objects, add bs.deriving abstract:

[@bs.deriving abstract]
type person = {
  name: string,
  age: int
};

bs.deriving abstract converts a record to an abstract type, and also creates a creation function:

let misha = person(~name="Misha", ~age=18);

Js.log(misha); /* => {"name":"Misha","age":18} */

You can also mark fields as optional with bs.optional and rename fields with bs.as.

Getters

bs.deriving abstract generates a getter function for every field.

Continuing the example above:

Js.log(nameGet(misha)); /* => "Misha" */
Js.log(ageGet(misha));  /* => undefined */

or, equivalently:

Js.log(misha->nameGet); /* => "Misha" */
Js.log(misha->ageGet);  /* => undefined */

Setters

Field values can't be updated unless marked with mutable.

bs.deriving abstract generates a setter function for every mutable field.

For example:

[@bs.deriving abstract]
type person = {
  name: string,
  [@bs.optional] mutable age: int,
  mutable country: string
};

let misha = person(~name="Misha", ~country="United States", ());

ageSet(misha, 18);
misha->countrySet("Australia");

compiles to:

var misha = {
  name: "Misha",
  country: "United States"
};

misha.age = 18;
misha.country = "Australia";

Note: bs.* must come before mutable.