@colisweb/rescript-toolkit 2.13.5 → 2.15.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@colisweb/rescript-toolkit",
3
- "version": "2.13.5",
3
+ "version": "2.15.0",
4
4
  "scripts": {
5
5
  "clean": "rescript clean",
6
6
  "build": "rescript build",
@@ -358,6 +358,143 @@ module UnitMeasure = {
358
358
  let convert = Toolkit__Utils_UnitMeasure.Time.makeH
359
359
  })
360
360
  }
361
+
362
+ module Currency = {
363
+ module WithUnit = {
364
+ let encoder: Decco.encoder<Toolkit__Utils_UnitMeasure.Currency.t> = value =>
365
+ switch value {
366
+ | #EUR(v) => Js.Json.string(v->Js.Float.toString ++ " EUR")
367
+ }
368
+
369
+ let decoder: Decco.decoder<Toolkit__Utils_UnitMeasure.Currency.t> = json =>
370
+ switch Decco.stringFromJson(json) {
371
+ | Ok(v) =>
372
+ let valueUnitArray = v->Js.String2.split(" ")
373
+ switch (
374
+ valueUnitArray[0]->Belt.Option.flatMap(Belt.Float.fromString),
375
+ valueUnitArray[1],
376
+ ) {
377
+ | (Some(v), Some("EUR")) => #EUR(v)->Ok
378
+ | (None, _) => Decco.error("unhandled no value", json)
379
+ | (_, None) => Decco.error("unhandled no unit", json)
380
+ | (_, Some(_)) => Decco.error("unhandled currency unit", json)
381
+ }
382
+
383
+ | Error(_) as err => err
384
+ }
385
+
386
+ let codec: Decco.codec<Toolkit__Utils_UnitMeasure.Currency.t> = (encoder, decoder)
387
+
388
+ @decco
389
+ type t = @decco.codec(codec) Toolkit__Utils_UnitMeasure.Currency.t
390
+ }
391
+
392
+ module type CurrencyConfig = {
393
+ let convert: float => Toolkit__Utils_UnitMeasure.Currency.t
394
+ }
395
+
396
+ module Make = (C: CurrencyConfig) => {
397
+ type currency = Toolkit__Utils_UnitMeasure.Currency.t
398
+
399
+ let encoder: Decco.encoder<Toolkit__Utils_UnitMeasure.Currency.t> = value =>
400
+ switch value {
401
+ | #EUR(v) => Js.Json.string(v->Js.Float.toString ++ " EUR")
402
+ }
403
+
404
+ let decoder: Decco.decoder<Toolkit__Utils_UnitMeasure.Currency.t> = json =>
405
+ switch Decco.floatFromJson(json) {
406
+ | Ok(v) => v->C.convert->Ok
407
+ | Error(_) as err => err
408
+ }
409
+
410
+ let codec: Decco.codec<currency> = (encoder, decoder)
411
+
412
+ let toValue = Toolkit__Utils_UnitMeasure.Currency.toValue
413
+ let display = Toolkit__Utils_UnitMeasure.Currency.toString
414
+
415
+ @decco
416
+ type t = @decco.codec(codec) Toolkit__Utils_UnitMeasure.Currency.t
417
+ }
418
+
419
+ module EUR = Make({
420
+ let convert = Toolkit__Utils_UnitMeasure.Currency.makeEUR
421
+ })
422
+ }
423
+
424
+ module CompositeUnits = {
425
+ module CurrencyPerDistance = {
426
+ module WithUnit = {
427
+ let encoder: Decco.encoder<
428
+ Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.t,
429
+ > = value =>
430
+ switch value {
431
+ | #EUR_km(v) => Js.Json.string(v->Js.Float.toString ++ " EUR/km")
432
+ }
433
+
434
+ let decoder: Decco.decoder<
435
+ Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.t,
436
+ > = json =>
437
+ switch Decco.stringFromJson(json) {
438
+ | Ok(v) =>
439
+ let valueUnitArray = v->Js.String2.split(" ")
440
+ switch (
441
+ valueUnitArray[0]->Belt.Option.flatMap(Belt.Float.fromString),
442
+ valueUnitArray[1],
443
+ ) {
444
+ | (Some(v), Some("EUR/km")) => #EUR_km(v)->Ok
445
+ | (None, _) => Decco.error("unhandled no value", json)
446
+ | (_, None) => Decco.error("unhandled no unit", json)
447
+ | (_, Some(_)) => Decco.error("unhandled currencyPerDistance unit", json)
448
+ }
449
+
450
+ | Error(_) as err => err
451
+ }
452
+
453
+ let codec: Decco.codec<Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.t> = (
454
+ encoder,
455
+ decoder,
456
+ )
457
+
458
+ @decco
459
+ type t = @decco.codec(codec) Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.t
460
+ }
461
+
462
+ module type CurrencyPerDistanceConfig = {
463
+ let convert: float => Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.t
464
+ }
465
+
466
+ module Make = (C: CurrencyPerDistanceConfig) => {
467
+ type currencyPerDistance = Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.t
468
+
469
+ let encoder: Decco.encoder<
470
+ Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.t,
471
+ > = value =>
472
+ switch value {
473
+ | #EUR_km(v) => Js.Json.string(v->Js.Float.toString ++ " EUR/km")
474
+ }
475
+
476
+ let decoder: Decco.decoder<
477
+ Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.t,
478
+ > = json =>
479
+ switch Decco.floatFromJson(json) {
480
+ | Ok(v) => v->C.convert->Ok
481
+ | Error(_) as err => err
482
+ }
483
+
484
+ let codec: Decco.codec<currencyPerDistance> = (encoder, decoder)
485
+
486
+ let toValue = Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.toValue
487
+ let display = Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.toString
488
+
489
+ @decco
490
+ type t = @decco.codec(codec) Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.t
491
+ }
492
+
493
+ module EUR_km = Make({
494
+ let convert = Toolkit__Utils_UnitMeasure.CompositeUnits.CurrencyPerDistance.makeEUR_km
495
+ })
496
+ }
497
+ }
361
498
  }
362
499
 
363
500
  // StringArray
@@ -64,21 +64,32 @@ let make = (
64
64
  placeholder=?{searchPlaceholder}
65
65
  onChange={event => {
66
66
  let target = event->ReactEvent.Form.currentTarget
67
-
68
- setSearch(_ => target["value"]->Js.String2.toLowerCase)
67
+ // normalize nfd -> replace by re remove split by commponents the accent letters and deletes the accent component only, leaving the un-accented letter
68
+ setSearch(_ =>
69
+ target["value"]
70
+ ->Js.String2.toLowerCase
71
+ ->Js.String2.normalizeByForm("NFD")
72
+ ->Js.String2.replaceByRe(%re("/[\u0300-\u036f]/g"), "")
73
+ )
69
74
  }}
70
75
  />
71
76
  </div>
72
77
  : React.null}
73
78
  {options
74
79
  ->Array.keep(({label}) =>
75
- search == "" || label->Js.String2.toLowerCase->Js.String2.includes(search)
80
+ // normalize nfd -> replace by re remove split by commponents the accent letters and deletes the accent component only, leaving the un-accented letter
81
+ search == "" ||
82
+ label
83
+ ->Js.String2.toLowerCase
84
+ ->Js.String2.normalizeByForm("NFD")
85
+ ->Js.String2.replaceByRe(%re("/[\u0300-\u036f]/g"), "")
86
+ ->Js.String2.includes(search)
76
87
  )
77
88
  ->Array.map(item => {
78
89
  let {itemLabel, label, value} = item
79
90
 
80
91
  <div
81
- key={label}
92
+ key={`multiselectoption-${label}-${value}`}
82
93
  className={cx(["flex flex-row items-center gap-2 mt-3 pt-3 text-left", itemClassName])}>
83
94
  <Toolkit__Ui_Checkbox
84
95
  value
@@ -1,5 +1,11 @@
1
1
  type size = [#md | #lg]
2
-
2
+ type color = [
3
+ | #info
4
+ | #success
5
+ | #primary
6
+ | #neutral
7
+ | #danger
8
+ ]
3
9
  let getBulletSize = size =>
4
10
  switch size {
5
11
  | #md => "w-5 h-5"
@@ -14,8 +20,8 @@ let getLabelSize = size =>
14
20
 
15
21
  let getTranslateBySize = size =>
16
22
  switch size {
17
- | #md => "translate-x-5"
18
- | #lg => "translate-x-8"
23
+ | #md => "translate-x-6"
24
+ | #lg => "translate-x-9"
19
25
  }
20
26
 
21
27
  @react.component
@@ -33,6 +39,7 @@ let make = (
33
39
  ~checkedSwitchClassName="",
34
40
  ~disabledSwitchClassName="",
35
41
  ~switchRef: option<ReactDOM.domRef>=?,
42
+ ~color=#primary,
36
43
  ) => {
37
44
  let (isChecked, setIsChecked) = React.useState(() => checked->Option.getWithDefault(false))
38
45
 
@@ -46,9 +53,17 @@ let make = (
46
53
  ])}>
47
54
  <span
48
55
  className={cx([
49
- "flex items-center relative rounded-full py-1 cursor-pointer border transition-all",
56
+ "flex items-center relative rounded-full py-[2px] cursor-pointer border transition-all",
50
57
  getLabelSize(size),
51
- isChecked ? "bg-info-500" : "bg-gray-300",
58
+ isChecked
59
+ ? switch color {
60
+ | #info => "bg-info-500 border-info-500"
61
+ | #success => "bg-success-500 border-success-500"
62
+ | #primary => "bg-primary-700 border-primary-700"
63
+ | #neutral => "bg-neutral-700 border-neutral-700"
64
+ | #danger => "bg-danger-500 border-danger-500"
65
+ }
66
+ : "bg-gray-300",
52
67
  switchClassName,
53
68
  isChecked ? checkedSwitchClassName : "",
54
69
  disabled->Option.getWithDefault(false) ? disabledSwitchClassName : "",
@@ -1,4 +1,11 @@
1
1
  type size = [#md | #lg]
2
+ type color = [
3
+ | #info
4
+ | #success
5
+ | #primary
6
+ | #neutral
7
+ | #danger
8
+ ]
2
9
 
3
10
  @react.component
4
11
  let make: (
@@ -15,4 +22,5 @@ let make: (
15
22
  ~checkedSwitchClassName: string=?,
16
23
  ~disabledSwitchClassName: string=?,
17
24
  ~switchRef: ReactDOM.domRef=?,
25
+ ~color: color=?,
18
26
  ) => React.element
@@ -103,3 +103,43 @@ module Dimension = {
103
103
  let makeM = (value: float) => #m(value)
104
104
  let makeKm = (value: float) => #km(value)
105
105
  }
106
+
107
+ module Currency = {
108
+ type t = [
109
+ | #EUR(float)
110
+ ]
111
+
112
+ let toValue = value =>
113
+ switch value {
114
+ | #EUR(v) => v
115
+ }
116
+
117
+ let toString = value => {
118
+ switch value {
119
+ | #EUR(v) => v->Float.toString ++ " EUR"
120
+ }
121
+ }
122
+
123
+ let makeEUR = (value: float) => #EUR(value)
124
+ }
125
+
126
+ module CompositeUnits = {
127
+ module CurrencyPerDistance = {
128
+ type t = [
129
+ | #EUR_km(float)
130
+ ]
131
+
132
+ let toValue = value =>
133
+ switch value {
134
+ | #EUR_km(v) => v
135
+ }
136
+
137
+ let toString = value => {
138
+ switch value {
139
+ | #EUR_km(v) => v->Float.toString ++ " EUR/km"
140
+ }
141
+ }
142
+
143
+ let makeEUR_km = (value: float) => #EUR_km(value)
144
+ }
145
+ }
@@ -127,3 +127,22 @@ external isFirstDayOfMonth: Js.Date.t => bool = "isFirstDayOfMonth"
127
127
 
128
128
  @module("date-fns")
129
129
  external getWeekNumber: (Js.Date.t, startOfWeekOptions) => int = "getWeek"
130
+
131
+ @module("date-fns")
132
+ external getDay: Js.Date.t => int = "getDay"
133
+
134
+ @module("date-fns")
135
+ external parse: string => Js.Date.t = "parse"
136
+
137
+ @module("date-fns")
138
+ external nextTuesday: Js.Date.t => Js.Date.t = "nextTuesday"
139
+ @module("date-fns")
140
+ external nextWednesday: Js.Date.t => Js.Date.t = "nextWednesday"
141
+ @module("date-fns")
142
+ external nextThursday: Js.Date.t => Js.Date.t = "nextThursday"
143
+ @module("date-fns")
144
+ external nextFriday: Js.Date.t => Js.Date.t = "nextFriday"
145
+ @module("date-fns")
146
+ external nextSaturday: Js.Date.t => Js.Date.t = "nextSaturday"
147
+ @module("date-fns")
148
+ external nextSunday: Js.Date.t => Js.Date.t = "nextSunday"
@@ -148,7 +148,7 @@ module ReactSelectMultipleCreateInput = {
148
148
  ~isLoading: option<bool>=?,
149
149
  ~className: option<string>=?,
150
150
  ~id: option<string>=?,
151
- ) => React.element = "CreatableSelect"
151
+ ) => React.element = "default"
152
152
  }
153
153
 
154
154
  @react.component
@@ -163,7 +163,7 @@ module ReactSelectMultipleCreateInput = {
163
163
  ~isLoading=false,
164
164
  ~className: option<string>=?,
165
165
  ~id: option<string>=?,
166
- ) =>
166
+ ) => {
167
167
  <ReactSelect
168
168
  components={"DropDownIndicator": React.null}
169
169
  isMulti=true
@@ -194,4 +194,5 @@ module ReactSelectMultipleCreateInput = {
194
194
  ?className
195
195
  ?id
196
196
  />
197
+ }
197
198
  }
@@ -0,0 +1,64 @@
1
+ type event<'a> = {
2
+ title: string,
3
+ start: Js.Date.t,
4
+ end: Js.Date.t,
5
+ resource: Js.t<'a>,
6
+ }
7
+
8
+ type view = [
9
+ | #month
10
+ | #week
11
+ | #work_week
12
+ | #day
13
+ | #agenda
14
+ ]
15
+
16
+ type eventProp<'a> = {
17
+ resource: Js.t<'a>,
18
+ start: Js.Date.t,
19
+ end: Js.Date.t,
20
+ isSelected: bool,
21
+ title: string,
22
+ }
23
+
24
+ type eventClassStyle
25
+
26
+ @obj
27
+ external makeEventClassStyle: (
28
+ ~className: string=?,
29
+ ~style: ReactDOM.Style.t=?,
30
+ unit,
31
+ ) => eventClassStyle = ""
32
+
33
+ @module("react-big-calendar") @react.component
34
+ external make: (
35
+ ~localizer: 'a,
36
+ ~events: array<event<'b>>,
37
+ ~defaultView: view=?,
38
+ ~views: array<view>=?,
39
+ ~culture: string=?,
40
+ ~timeslots: int=?,
41
+ ~min: Js.Date.t=?,
42
+ ~max: Js.Date.t=?,
43
+ ~step: int=?,
44
+ ~components: 'z=?,
45
+ ~eventPropGetter: eventProp<'b> => eventClassStyle,
46
+ ~messages: Js.t<'message>=?,
47
+ ~formats: Js.t<'formats>=?,
48
+ ~onSelectEvent: event<'b> => unit=?,
49
+ ~onView: view => unit=?,
50
+ ~onNavigate: Js.Date.t => unit=?,
51
+ ) => React.element = "Calendar"
52
+
53
+ type calendarLocalizer
54
+
55
+ @module("react-big-calendar")
56
+ external dateFnsLocalizer: 'a => calendarLocalizer = "dateFnsLocalizer"
57
+
58
+ module Constants = {
59
+ @module("react-big-calendar/lib/utils/constants") @scope("navigate")
60
+ external previous: string = "PREVIOUS"
61
+
62
+ @module("react-big-calendar/lib/utils/constants") @scope("navigate")
63
+ external next: string = "NEXT"
64
+ }