mobx
6.13.5
Patch Changes
6.13.4
Patch Changes
6.13.3
Patch Changes
6.13.2
Patch Changes
6.13.1
Patch Changes
6.13.0
Minor Changes
6.12.5
Patch Changes
6.12.4
Patch Changes
6.12.2
Patch Changes
61abc53f
#3852 Thanks @mweststrate! - Patched the release process, forcing release to get everything in pristine state.
b28e0ebb
#3816 Thanks @barroij! - Fix IReactionDisposer
and IIsObservableObject
interface definition so that Typescript knows the property key $mobx
is a symbol and not a string
6.12.1
Patch Changes
6.12.0
Minor Changes
ec5db592
#3792 Thanks @tonyraoul! - Improve observablearray proxy pefromance for es2023.array and es2022.array methods
Patch Changes
6.11.0
Minor Changes
6.10.2
Patch Changes
c8d9374d
#3748 Thanks @mweststrate! - Fixed: #3747, computed values becoming stale if the underlying observable was created and updated outside a reactive context
6.10.1
Patch Changes
3ceeb865
#3732 Thanks @urugator! - - fix: #3728: Observable initialization updates state version.
- fix: Observable set initialization violates
enforceActions: "always"
.
- fix: Changing keys of observable object does not respect
enforceActions
.
6.10.0
Minor Changes
bebd5f05
#3727 Thanks @rluvaton! - Added support for signal
(AbortSignal) in autorun
, reaction
and sync when
options to dispose them
Patch Changes
6.9.1
Patch Changes
6.9.0
Minor Changes
44a2cf42
#3590 Thanks @urugator! - Better support for React 18: Mobx now keeps track of a global state version, which updates with each mutation.
6.8.0
Minor Changes
fed3ff14
#3608 Thanks @emereum! - Do not expose the methods observe_
or intercept_
on computed values and observable values.
Patch Changes
6.7.0
Minor Changes
Patch Changes
6.6.2
Patch Changes
6.6.1
Patch Changes
6.6.0
Minor Changes
6.5.0
Minor Changes
6.4.2
Patch Changes
2caf7e1a
#3316 Thanks @urugator! - requiresObservable
always takes precedence over global reactionRequiresObservable
6.4.1
Patch Changes
6.4.0
Minor Changes
6b926833
#3287 Thanks @kubk! - Remove any from 'merge' and 'replace' methods of ObservableMap
Patch Changes
dee35be4
#3300 Thanks @kubk! - Make the error message about Object.freeze more informative
6.3.13
Patch Changes
6.3.12
Patch Changes
6.3.11
Patch Changes
6.3.10
Patch Changes
6.3.9
Patch Changes
6.3.8
Patch Changes
6.3.7
Patch Changes
6.3.6
Patch Changes
6.3.5
Patch Changes
6.3.4
Patch Changes
6.3.3
Patch Changes
6.3.2
Patch Changes
6.3.1
Patch Changes
6.3.0
Minor Changes
Patch Changes
6.2.0
Minor Changes
bc8db3df
#2779 Thanks @urugator! - In mobx5 all own properties were by default observable regardless of their value. Since mobx6 we treat functional properties as action
s or to be precise autoAction
s. autoAction
provides action
's benefits to your functions, without the need to explicitely annotate them as actions
.
We think this is useful, but as a consequence, such properties are no longer observable
and therefore non-writable and also non-enumerable. This turned out to be suprising and inconvinient to some users:
https://github.com/mobxjs/mobx/discussions/2760
https://github.com/mobxjs/mobx/discussions/2586
https://github.com/mobxjs/mobx/issues/2835
https://github.com/mobxjs/mobx/issues/2629
https://github.com/mobxjs/mobx/issues/2551
https://github.com/mobxjs/mobx/issues/2637
So we decided to change it: All own props including functions are observable
(enumerable, writable) as in v5, but additionally all functions that become part of deeply observable structure are by default converted to autoAction
/flow
.
Note that deep
option affects this conversion in the same way as it affects conversion of other values (object/array/map/set).
- by default all functions are converted to
autoAction
s/flow
s
- by default all originally own props are now observable and enumerable (as in pre v6)
deep: false
ignores all property values (including functions that would be previously converted to autoAction
/flow
)
- by default lone setters are converted to
action
s
Patch Changes
6.1.8
Patch Changes
6.1.7
Patch Changes
5640aa77
#2799 Thanks @urugator! - - fix: user provided debug names are not preserved on production
- fix: property atom's debug name is dynamic on production
- fix:
observable(primitive, options)
ignores options
- fix:
getDebugName(action)
throws [MobX] Cannot obtain atom from undefined
- fix: terser using
unsafe: true
6.1.6
Patch Changes
9b195b17
#2780 Thanks @iChenLei! - The overall memory usage of MobX has been reduced in production builds by skipping the generation of debug identifiers. The internal mapid_
field of Reaction has been removed as part of the change.
6.1.5
Patch Changes
6.1.4
Patch Changes
6.1.3
Patch Changes
6.1.2
Patch Changes
ca09f2f5
#2761 Thanks @urugator! - configure({ safeDescriptors: false })
now forces all props of observable objects to be writable and configurable
6.1.1
Patch Changes
39eca50d
#2757 Thanks @urugator! - Fix error stringification on minified build
Fix isObservableProp
not supporting Symbols
Fix makeAutoObservable
not ignoring inferredAnnotationsSymbol
6.1.0
This release fixes a plethora of bugs related to sub-classing and reflecting / iterating on observable objects.
The behavior of MobX is in many edge cases much more explicitly defined now.
A new annotation was introduced: @override
/ override
to support re-defining actions and computed values(!) on sub classses.
For idiomatic MobX usage this release should have little impact, but if you are using a lot of sub-classing, reflection APIs or direct object manipulations like defineProperty
,
this release might introduce previously unseen errors for cases that silently failed before, or even worked successfully even though the correct behavior wasn't specified earlier.
If you are migrating from MobX 4/5 we strongly recommend to go to 6.1 in one go, and skip 6.0.*, as some buggy behavior compared to the previous majors has been corrected.
As always, our libraries come as-is and are maintained by volunteers. Upgrades are at own risk and voluntary. Bug reports require a minimal reproductions and a correctly filled out issue template.
Support the ongoing maintenance at: https://opencollective.com/mobx
Minor Changes
28f8a11d
#2641 Thanks @urugator!
action
, computed
, flow
defined on prototype can be overridden by subclass via override
annotation/decorator. Previously broken.
- Overriding anything defined on instance itself (
this
) is not supported and should throw. Previously partially possible or broken.
- Attempt to re-annotate property always throws. Previously mostly undefined outcome.
- All annotated and non-observable props (action/flow) are non-writable. Previously writable.
- All annotated props of non-plain object are non-configurable. Previously configurable.
- Observable object should now work more reliably in various (edge) cases.
- Proxied objects now support
Object.defineProperty
. Previously unsupported/broken.
extendObservable/makeObservable/defineProperty
notifies observers/listeners/interceptors about added props. Previously inconsistent.
keys/values/entries
works like Object.keys/values/entries
. Previously included only observables.
has
works like in
. Previously reported true
only for existing own observable props.
set
no longer transforms existing non-observable prop to observable prop, but simply sets the value.
remove/delete
works with non-observable and computed props. Previously unsupported/broken.
- Passing
options
to observable/extendObservable/makeObservable
throws if the object is already observable . Previously passed options were mostly ignored.
autoBind
option is now sticky - same as deep
and name
option.
observable/extendObservable
now also picks non-enumerable keys (same as make[Auto]Observable
).
- Removed deprecated
action.bound("name")
- Proxied objects should be compatible with
Reflect
API. Previously throwing instead of returning booleans.
6.0.5
Patch Changes
🚨🚨🚨 If you are upgrading from V4/V5, please follow the migration guide.
6.0.4
Patch Changes
6.0.3
Patch Changes
d0e6778d
- Create ESM bundles with NODE_ENV correctly replaced so it can be used in browser (#2564)
6.0.2
Patch Changes
6.0.1
- Fixed issue in TS typings of
makeObservable
in combination with a member named toString()
6.0.0
New features
makeObservable(target, annotations)
is now the recommended way to make objects with a fixed shape observable, such as classes.
makeAutoObservable(target)
will automatically determine the annotations used by makeObservable
. Methods will be marked as 'autoAction', so that they can be used both from a computed value or as standalone method.
- MobX 6 can be used in both modern environments, and environments that don't support Proxy. So both MobX 4 and 5 users can upgrade to 6. See proxy support for more details.
observable.array
now supports { proxy: false }
as option.
reaction
's effect function now receives the previous value seen by the reaction as second argument.
flow
can now be used as annotation as well. You might need flowResult
in case you use TypeScript to extract the correct result type. details.
Breaking changes
Changes that might affect you
- The
decorate
API has been removed, and needs to be replaced by makeObservable
in the constructor of the targeted class. It accepts the same arguments. The mobx-undecorate
can transform this automatically.
- When using
extendObservable
/ observable
, fields that contained functions used to be turned into observables. This is no longer the case, they will be converted into autoActions
.
- Strict mode for actions is now enabled by default in
observed
mode.
toJS
no longer takes any options. It no longer converts Maps and Sets to plain data structures. Generic, flexible serialization of data structures is out of scope for the MobX project, and writing custom serialization methods is a much more scalable approach to serialization (tip: leverage computed
s to define how class instances should be serialized).
- The methods
intercept
and observe
are no longer exposed on observable arrays, maps and boxed observables. Import them as utility from mobx instead: import { observe, intercept } from "mobx"
, and pass the collection as first argument: observer(collection, callback)
. Note that we still recommend to avoid these APIs.
observableMap.toPOJO()
, observableMap.toJS()
have been dropped. Use new Map(observableMap)
instead if you want to convert an observable map to a plain Map shallowly.
observableMap.toJSON()
now returns an entries array rather than a new Map, to better support serialization.
observableSet.toJS()
has been dropped. Use new Set(observableSet)
instead if you want to convert an observable Set to a plain Set shallowly.
observableSet.toJSON()
now returns an array rather than a new Set, to better support serialization.
- Sorting or reversing an observableArray in a derivation (without slicing first) will now throw rather than warn. In contrast, it is now allowed to sort or reverse observable arrays in-place, as long as it happens in an action.
isArrayLike
is no longer exposed as utility. Use Array.isArray(x) || isObservableArray(x)
instead.
Obscure things that don't work anymore, but that probably won't affect you
- It is no longer possible to re-decorate a field (through either
@observable
or makeObservable
) that is already declared in a super class.
runInAction
no longer supports passing a name as first argument. Name the original function or use action(name, fn)()
if you care about the debug name.
computed(getterFn, setterFn)
no longer accepts a setter function as a second argument. Use the set
option instead: computed(getterFn, { set: setterFn })
.
- In observable arrays, for
findIndex
/ find
method, the offset
argument (the third one) is no longer supported, to be consistent with ES arrays.
- The option
computedConfigurable
of configure
is no longer supported as it is now the default.
observableArray.toJS()
has been removed, use observableArray.slice()
instead, which does the same.
- Killed support for the
IGNORE_MOBX_MINIFY_WARNING
environment flag.
_allowStateChangesInComputation(fn)
is no longer needed, use runInAction(fn)
instead.
- In
computed
, the when
predicate (first arg), and reaction
predicate (first arg) it is now forbidden to directly change state. State changes should be done in their effect functions, or otherwise at least wrapped in runInAction
(only the state change, not the observables you want to track!). Note that this is still an anti-pattern.
- The
observableArray.get()
and observableArray.set()
methods are no longer supported.
- The
IObservableObject
interface is no longer exported from MobX.
- The second argument to the
reaction
effect function, the disposer object, is now passed in as third argument. The second argument is now the previous value seen by the reaction.
onBecomeObserved
/ onBecomeUnobserved
will now only trigger for observables that are actually used by a reaction (see #2309 for background).
Fixes
- #2326: Incorrect
this
for array callbacks such as in array.forEach
- #2379: Fixed issue with
array.concat
- #2309: Fixed several inconsistencies between keepAlive'd computed values and
on(un)BecomeObserved
- Fixed several inconsistencies when
on(un)BecomeObserved
was triggered for observables changed in actions without having an observer
5.15.7 / 4.15.7
5.15.6 / 4.15.6
5.15.5 / 4.15.5
- Fixed ObservableSet.prototype.forEach not being reactive in 4.x #2341 by @melnikov-s
- Add error when computed value declared for unspecified getter #1867 by @berrunder
- Fixed #2326 correct array is passed to callbacks by @urugator
- Fixed #2278,
map.delete
should respect strict mode, by @urugator
- Fixed #2253, #1980, map key order not always preserved by @urugator
Fixed #2412, non-enumerable getters not being picked up by extendObservable
by @urugator
Several performance improvements
- Dropped
browser
fields from package.json
5.15.4 / 4.15.4
5.15.3 / 4.15.3
5.15.2 / 4.15.2
5.15.1 / 4.15.1
- Make initial values of observable set accept readonly array #2202
- Expose
_allowStateReadsStart
& _allowStateReadsEnd
. This is low level stuff you shouldn't need that's mostly useful for library creators. #2233
- Fixed an issue with
observableRequiresReaction
and updating observable during reaction #2195
- Improved type inference for
action
#2213 (see detailed explanation)
5.15.0
The minimum required TypeScript version is now 3.6
- Fixed flow typings with Typescript v3.6. This means that version of Typescript is required when using flows.
- Cancelled flows now reject with a
FlowCancellationError
instance whose error message is the same as in previous versions ("FLOW_CANCELLED"
) so this is not breaking. #2172 by @vonovak
- Fix running mobx in web worker #2184 by @shahata
- Fixed flow typings for Facebook's Flow. A new
CancellablePromise
Flow type is exported. #2164 by @vonovak
- Added support for symbol keys on observable properties (MobX 5 only). #2175 by @StephenHaney
5.14.2
- Fixed installation issue trying to run
postinstall
hook for a website #2165.
5.14.1 / 4.14.1
- Fixed a possible issue with action stack errors and multiple mobx versions installed at the same time #2135.
- Added
comparer.shallow
for shallow object/array comparisons #1561.
- Fixed disposing an interception within an interception throwing an error #1950.
5.14.0 / 4.14.0
- Added experimental
reactionRequiresObservable
& observableRequiresReaction
config #2079, Docs
- Added experimental
requiresObservable
config to reaction
, autorun
& when
options #2079, Docs
5.13.1 / 4.13.1
- Don't use
global
and self
keywords unless defined. Fixes #2070.
- onBecome(Un)Observed didn't trigger when using number as key of observable map. Fixes #2067.
- Exposed
_startAction
and _endAction
to be able to start and action and finish it without needing a code block. This is low level stuff you shouldn't need that's mostly useful for library creators.
5.13.0 / 4.13.0
- Fixed potential memory leak in observable maps, when non-primitive values are used as keys. Fixes #2031 through #2032.
- Added support to store additional non-observable(!) fields (string or symbol based) on array, to better reflect behavior of MobX 4. Fixes #2044 through #2046
5.11.0 / 4.12.0
4.11.0
BREAKING CHANGE
Reverted the support of Symbols in general in MobX 4, as it gives to many potential build errors and increases the system requirements for MobX 4 (which was an oversight in 4.10.0). Apologies for the breaking change (lack of new major version numbers). If lock files are properly used however, no harm should be caused by this change.
- Reverted
Symbol
support in observable maps and objects. Reverts #1944 through #1988. Fixes #1986, #1987
5.10.1
- Fixed a recent regression where array update events would send undefined as
change.object
through #1985 by xaviergonz
5.10.0 / 4.10.0
5.9.4 / 4.9.4
- Allow symbol keys in
ObservableMap
, see #1930 by pimterry
- Fixed type definitions of
toStringTag
for Maps and Sets, see #1929 by lennerd
4.9.3
- Fixed
observable.set
compatibility with IE 11, see #1917 by kalmi
4.9.2
- Fixed regression #1878, accidental use of
Symbol
breaking Internet Explorer / React Native compatibility.
4.9.1
- Fixed regression in
toJS
: observable maps were not properly serialized. Fixes #1875
5.9.0 / 4.9.0
Features
- Introduced support for observable sets! Through #1592 by @newraina
observable.box
now accepts an equals
option, to be able to pass a custom comparision function. Through #1862, #1874 by @fi3ework. Fixes #1580
- Improved logging of reactions; if an action throws an exception, errors in reactions that react to that are only logged as warnings. Fixes #1836
Fixes
- Improved typings for
flow
, see #1827 by @xaviergonz
- Don't allow subclassing map, fixes #1858
- Fixed
trace(true)
not being able to handle multi-line comments in traced function. Fixes #1850
@computed
now introduces non-configurable properties, to fail fast on incorrect inheritance or property deletion. Fixes #1867
- The options
enforceActions
and isolateGlobalState
now work correctly when used together. Fixes #1869
5.8.0 / 4.8.0
- MobX now requires TypeScript 3 (this was already the case in 5.7.0, but in this version the difference is actually noticeable in the typings).
- Fixed array dehancer sometimes skipping. Fixes #1839 through #1841 by k-g-a
- Fixed issue where webpack 4 wouldn't use the ESM module #1834 by mrtnbroder
- Improved type inference for
flow
in TypeScript 3. Fixes #1816 through #1825 by ismailhabib
- Introduced support for global environment variable
IGNORE_MOBX_MINIFIY_WARNING=true
to skip the built-in minification warning. See #1835 by fi3ework
- Fixed onBecome(Un)Observed dispoer cleanup. Fixes #1537 through #1833 by fi3ework
5.7.1 / 4.7.1
- Fixed #1839, ObservableArrayAdministration.dehanceValues does not dehance last value.
5.7.0 / 4.7.0
- Upgraded typings to TypeScript 3
- Fixed #1742, change detection fails when multiple mobx instances were active.
- Fixed #1624, use built-in flow types for iterators
- Fixed #1777 through #1826, stack overflow exception, in development mode, when using
@computed
on a React component. The MobX 5 behavior here has been reverted to the MobX 4 behavior.
5.6.0 / 4.6.0
keepAlive
has become smarter and won't recomputed computed values that are kept alive, as long as they aren't read. Implements #1534
- Fixed #1796, undeleting a property that had an initial value of
undefined
was undetected
- Improved Flow typings, see #1794 and #1786
5.5.2 / 4.5.2
5.5.1 / 4.5.1
toJS
now has a recurseEverything
everything option, that even detects and converts observable objects that are "behind" non-observable objects. See #1699 by wangyiz4262
- Added flow typings form
comparer
, see #1751 by pdong
- Update flow typings for configuration options, #1772 by alexandersorokin
5.5.0 / 4.5.0
(Minor version of 5
was bumped significantly to make the number better correlate together :-))
- Fixed #1740: combining decorators and
extendObservable
in a class constructor caused errors to be thrown
- Fixed #1739:
- Proxies:
delete
-ing a property was not always picked up by the reactivity system
- Non-proxies:
remove()
-ing a property was not always picked up by the has()
and get()
utilities
has
now returns true
for computed fields
get
now returns a value for computed fields
- Introduced
_allowStateChangeInsideComputed
. Don't use it :-).
- MobX is now transpiled using babel 7
5.1.2 / 4.4.2
- Fixed #1650, decorating fields with the name
toString
does not behave correctly.
5.1.1 / 4.4.1
5.1.0 / 4.4.0
- Improved handling of multiple MobX instances. MobX will no longer complain about being loaded multiple times (one should still prevent it though, to prevent blowing up bundle size!), but merge internal state by default. If multiple MobX versions need to be loaded, call
configure({ isolateGlobalState: true })
. Note that this means that observables from the different MobX instances won't cooperate. Fixes #1681, #1082
enforceActions
options now supports the values: "never"
, "observed"
and "always"
to make the behavior more clear. Fixes #1680, #1473
5.0.5
- Fixed #1667: creating a large array could result in undefined items (MobX 4.* was not affected)
4.3.2 / 5.0.4
- Fixed #1685: expose
IAutorunOptions
decorate
now can apply multiple decorators, by accepting an array and applying them right to left: decorate(Todo, { title: [serializable(primitive), persist('object'), observable] })
. By @ramybenaroya through #1691 and #1686
- Improved typings of
flow
so that it accepts async generators. By @dannsam through #1656 and #1655
keys()
now also supports arrays. Fixes #1600 through #1601 by @nunocastromartins
5.0.3
- Fixed issue where it was no longer possible to define custom properties on observable arrays
5.0.2
- Fixed issue where iterators where not compiled to ES5, breaking the ES5 based builds.
5.0.1 (Unpublished)
- Fixed regression bug:
ObservableMap.size
was no longer observable. Fixes #1583
- Downgraded lib export from ES6 to ES5. To many build tools still trip over ES6. Fixes #1584. A modern build is available through
import ... from "mobx/lib/mobx.es6"
(or setup an alias in your build system)
- Added support for mobx-react-devtools
5.0.0
Release blogpost
Proxy support!
MobX 5 is the first MobX version fully leveraging Proxies. This has two big advantages
- MobX can now detect the addition of properties on plain observable objects, so it is now possible to use plain observable objects as dynamic collections.
- Observable arrays are now recognized as arrays by all third party libraries, which will avoid the need to slice them.
The system requirements to run MobX has been upped
- MobX 5 can only be used on environments that support
Proxies
. See the browser support for details.
- Since MobX no longer runs on older browsers, the compilation target has been upgraded to ES2015 syntax supporting browsers. This means that MobX is not loadable on older browsers without down compilation to ES5.
- If for whatever reason your project cannot meet this requirements, please stick to MobX 4. It will be actively maintained. All current features of MobX 5 are expressable in MobX 4 as well, but it means that for example to use dynamic objects some additional APIs are needed.
- The performance footprint of MobX 5 should be pretty similar to MobX 4. In our performance tests we saw some minor improvements in memory footprint, but overall it should be pretty comparable.
Breaking changes
- The required runtime needs to support the non-polyfillable
Proxy
API.
- The minimum runtime target is now ES2015, not ES5
spy
has become a no-op in production builds
- All earlier deprecated APIs are dropped. Make sure to not have any deprecation warnings before upgrading.
array.move
and array.peek
are removed from the API
- Dropped the third argument to
array.find
and array.findIndex
since they were not standardized in ES.
.$mobx
property has been dropped from all observables and replaced by a Symbol. Instead of using x.$mobx.name
, use import { $mobx } from "mobx"; x[$mobx].name
etc.
- In some cases, the order in which autoruns are fired could have changed due to some internal optimizations (note that MobX never had a guarantee about the order in which autoruns fired!)
New features
- It is possible to pass the
proxy: false
argument to observable.object
to disable proxying (theoretically slightly faster, but removes no dynamic key addition)
Known Issues
- Observable objects can no longer be frozen (otherwise they would become un-observable😎). If you are actually trying to do so MobX will now throw an exception like:
[mobx] Dynamic observable objects cannot be frozen]
. A place where that might happen unexpectedly is when passing an observable object as style
property to a React component. Like <span style={someObservableObject} />
, since React will freeze all style objects. The work-around is to simply pass a fresh, non-observable object for styling like: <span style={{...someObservableObject}} />
.
If you are using mobx
with mobx-react
, and you are upgrading mobx-react
to the MobX 5 compatible version (`mobx-react@5.2.0) you will notice that
this.propsor
this.stateare not yet observable in the
constructoror
componentWillMount. This is for forward compatibility with React 16.3 where
componentWillMounthas been deprecated. In most cases using
componentDidMount` instead will suffice, especially when the goal is to setup reactions. For more info see #478. Fixed in mobx-react 5.2.1. But note that you should still migrate away from componentWillMount
😎.
- Jest
toEqual
might throw an error allKeys[x].match is not a function
when trying to equal observable arrays. This is a bug in Jest report. The simple work around for now is to slice (or toJS
if the problem is recursive) the array first.
- Jest
toEqual
matcher might no longer correctly equal your class instances, complaining about differences in the MobX adminstration. This is due to a bug with the processing of symbols: report. For now you might want to use a custom matcher if you are directly equalling observable objects. As a work around toJS(object)
could be used before diffing.
Note June 7th, 2018: Both issues are already in Jest master and should be released soon.
Migration guide
- Make sure to not use any API that produces deprecation warnings in MobX 4. Beyond that MobX 5 should pretty well as drop-in replacement of MobX 4.
- You could perform the following clean ups:
- Don't
slice()
arrays when passing them to external libraries. (Note you still shouldn't pass observable data structures to non-observer
React components, which is an orthogonal concept)
- You could replace observable maps with observable objects if you are only using string-based keys.
- Don't call the
reverse
or sort
operations directly on observableArray's anymore, as it's behavior slightly differed from the built-in implementations of those methods. Instead use observableArray.slice().sort()
to perform the sort on a copy. This gives no additional performance overhead compared to MobX 4. (The reason behind this is that built-in sort
updates the array in place, but the observable array implementation always performed the sort on a defensive copy, and this change makes that explicit).
- you may remove usages of
isArrayLike()
since Array.isArray()
will now return true for observable arrays
API's that have been dropped
- The
arrayBuffer
setting is no longer supported by configure
(it has become irrelevant)
observable.shallowBox
, observable.shallowArray
, observable.shallowMap
, observable.shallowObject
, extendShallowObservable
api's have been removed. Instead, pass { deep: false }
to their non-shallow counter parts.
observableArray.peek
, observableArray.move
4.3.1
4.3.0
- Introduced the
entries(observable)
API, by @samjacobclift through #1536
- Fixed #1535: Change in nested computed value was not propagated if read outside action context when there is a pending reaction. For more details see the exact test case.
- Illegal property access through prototypes is now a warning instead of an error. Fixes #1506. By @AmazingTurtle through #1529
- Fixed issue where providing a custom setter to
@computed({ set: ... })
wasn't picked up
- Fixed #1545: Actions properties where not re-assignable when using TypeScript
- Illegal Access checks are now a warning instead of an error. Fix
4.2.1
- Fixed flow typings for
mobx.configure
#1521 by @andrew--r
- Improved typings for
mobx.flow
, fixes #1527
- Throw error when using
@observable
in combination with a getter. #1511 by @quanganhtran
toJS
now uses Map internally, for faster detection of cycles. #1517 by @loatheb
- Fixed #1512:
observe
hooks not being triggered when using mobx.set
, Fixed in #1514 by @quanganhtran
- Several minor improvements, additional tests and doc improvements.
4.2.0
- Introduced
configure({ enforceActions: "strict" })
, which is more strict then enforceActions: true
, as it will also throw on non-observed changes to observables. See also #1473
- Fixed #1480: Exceptions in the effect handler of
reaction
where not properly picked up by the global reaction system
- Fixed a bug where computed values updated their cached value, even when the comparer considered the new value equal to the previous one. Thanks @kuitos for finding this and fixing it! #1499
- Undeprecated
ObservableMap
, fixes #1496
- Observable arrays now support
Symbol.toStringTag
(if available / polyfilled). This allows libraries like Ramda to detect automatically that observable arrays are arrays. Fixes #1490. Note that Array.isArray
will keep returning false for the entire MobX 4 range.
- Actions are now always
configurable
and writable
, like in MobX 3. Fixes #1477
- Merged several improvements to the flow typings. #1501 by @quanganhtran
- Fixed several accidental usages of the global
fail
, by @mtaran-google through #1483 and #1482
4.1.1
- Import
default
from MobX will no longer throw, but only warn instead. This fixes some issues with tools that reflect on the default
export of a module
- Disposing a spy listener inside a spy handler no longer causes an exception. Fixes #1459 through #1460 by farwayer
- Added a missing
runInAction
overload in the flow typings. #1451 by AMilassin
- Improved the typings of
decorate
. See #1450 by makepost
4.1.0
- Introduced
keepAlive
as option to computed
- All observable api's now default to
any
for their generic arguments
- Improved
flow
cancellation
- The effect of
when
is now automatically an action.
@computed
properties are now declared on their owner rather then the protoptype. Fixes an issue where @computed
fields didn't work in React Native on proxied objects. See #1396
action
and action.bound
decorated fields are now reassignable, so that they can be stubbed
4.0.2
- Fixed issue where exceptions like
TypeError: Cannot define property:__mobxDidRunLazyInitializers, object is not extensible.
were thrown. Fixes #1404
- Improved flow typings for
flow
, #1399 by @ismailhabib
4.0.1
4.0.0
- For the highlights of this release, read the blog:
- For migration notes: see the wiki page
- Note; many things that were removed to make the api surface smaller. If you think some feature shouldn't have been removed, feel free to open an issue!
This is the extensive list of all changes.
New features
The changes mentioned here are discussed in detail in the release highlights, or were simply updated in the docs.
- MobX 4 introduces separation between the production and non production build. The production build strips most typechecks, resulting in a faster and smaller build. Make sure to substitute process.env.NODE_ENV = "production" in your build process! If you are using MobX in a react project, you most probably already have set this up. Otherwise, the idea is explained here.
- Introduced
flow
to create a chain of async actions. This is the same function as asyncActions
of the mobx-utils package
- These
flow
's are now cancellable, by calling .cancel()
on the returned promise, which will throw a cancellation exception into the generator function.
flow
also has experimental support for async iterators (async * function
)
- Introduced
decorate(thing, decorators)
to decorate classes or object without needing decorator syntax.
- Introduced
onBecomeObserved
and onBecomeUnobserved
. These API's enable hooking into the observability system and get notified about when an observable starts / stops becoming used. This is great to automaticaly fetch data from external resources, or stop doing so.
computed
/ @computed
now accepts a requiresReaction
option. If it set, the computed value will throw an exception if it is being read while not being tracked by some reaction.
- To make
requiresReaction
the default, use mobx.configure({ computedRequiresReaction: true })
- Introduced
mobx.configure({ disableErrorBoundaries })
, for easier debugging of exceptoins. By NaridaL through #1262
toJS
now accepts the options: { detectCycles?: boolean, exportMapsAsObjects?: boolean }
, both true
by default
- Observable maps are now backed by real ES6 Maps. This means that any value can be used as key now, not just strings and numbers.
The flow typings have been updated. Since this is a manual effort, there can be mistakes, so feel free to PR!
computed(fn, options?)
/ @computed(options) get fn()
now accept the following options:
set: (value) => void
to set a custom setter on the computed property
name: "debug name"
equals: fn
the equality value to use for the computed to determine whether its output has changed. The default is comparer.default
. Alternatives are comparer.structural
, comparer.identity
or just your own comparison function.
requiresReaction: boolean
see above.
autorun(fn, options?)
now accepts the following options:
delay: number
debounce the autorun with the given amount of milliseconds. This replaces the MobX 3 api autorunAsync
name: "debug name"
scheduler: function
a custom scheduler to run the autorun. For example to connect running the autorun to requestAnimationFrame
. See the docs for more details
onError
. A custom error handler to be notified when an autorun throws an exception.
reaction(expr, effect, options?)
now accepts the following options:
delay: number
debounce the autorun with the given amount of milliseconds. This replaces the MobX 3 api autorunAsync
fireImmediately
. Immediately fire the effect function after the first evaluation of expr
equals
. Custom equality function to determine whether the expr
function differed from its previous result, and hence should fire effect
. Accepts the same options as the equals
option of computed.
- All the options
autorun
accepts
when(predicate, effect?, options?)
now accepts the following options:
name: "debug name"
onError
. A custom error handler to be notified when an autorun throws an exception.
timeout: number
a timeout in milliseconds, after which the onError
handler will be triggered to signal the condition not being met within a certain time
The effect
parameter of when
has become optional. If it is omitted, when
will return a promise. This makes it easy to await
a condition, for example: await when(() => user.profile.loaded)
. The returned promise can be cancelled using promise.cancel()
There is now an utility API that enables manipulating observable maps, objects and arrays with the same api. These api's are fully reactive, which means that even new property declarations can be detected by mobx if set
is used to add them, and values
or keys
to iterate them.
values(thing)
returns all values in the collection as array
keys(thing)
returns all keys in the collection as array
set(thing, key, value)
or set(thing, { key: value })
Updates the given collection with the provided key / value pair(s).
remove(thing, key)
removes the specified child from the collection. For arrays splicing is used.
has(thing, key)
returns true if the collection has the specified observable property.
get(thing, key)
returns the chlid under the specified key.
observable
, observable.array
, observable.object
, observable.map
and extendObservable
now accept an additional options object, which can specify the following attributes:
name: "debug name"
deep: boolean
. true
by default, indicates whether the children of this collection are automatically converted into observables as well.
defaultDecorator: <decorator>
specifies the default decorator used for new children / properties, by default: observable.deep
, but could be changed to observable.ref
, observable.struct
etc. (The deep
property is just a short-hand for switching between observable.deep
or observable.ref
as default decorator for new properties)
Breaking changes
The changes mentioned here are discussed in detail in the migration notes
- MobX 4 requires
Map
to be globally available. Polyfill it if targeting IE < 11 or other older browsers.
- For typescript users, MobX now requires
Map
and several Symbol
s to exist for its typings. So make sure that the lib
configuration of your project is set to "es6"
. (The compilation target can still be "es5"
)
observable.shallowArray(values)
has been removed, instead use observable.array(values, { deep: false })
observable.shallowMap(values)
has been removed, instead use observable.map(values, { deep: false })
observable.shallowObject(values)
has been removed, instead use observable.object(values, {}, { deep: false })
extendShallowObservable(target, props)
, instead use extendObservable(target, props, {}, { deep: false })
- The decorators
observable.ref
, observable.shallow
, observable.deep
, observable.struct
can no longer be used as functions. Instead, they should be passed as part of the decorators
param to resp. observable.object
and extendObservable
- The new signature of
extendObservable
is extendObservable(target, props, decorators?, options?)
. This also means it is no longer possible to pass multiple bags of properties to extendObservable
. extendObservable
can no longer be used to re-declare properties. Use set
instead to update existing properties (or introduce new ones). Update 13-01-2020: the latter limitation has been reverted in MobX 4.15.2 / 5.15.2
- Iterating maps now follows the spec, that is,
map.values()
, map.entries()
, map.keys()
, map[@@iterator]()
and array[@@iterator]()
no longer return an array, but an iterator. Use mobx.values(map)
or Array.from(map)
to convert the iterators to arrays.
- dropped `@computed.equals
, instead, you can now use
@computed({ equals: ... })`
useStrict(boolean)
was dropped, use configure({ enforceActions: boolean })
instead
isolateGlobalState
was dropped, use configure({ isolateGlobalState: true})
instead
- If there are multiple mobx instances active in a single project, an exception will be thrown. Previously only a warning was printed. Fixes #1098. For details, see #1082.
- Dropped the
shareGlobalState
feature. Instead, projects should be setup properly and it is up to the hosting package to make sure that there is only one MobX instance
expr
has been moved to mobx-utils. Remember, expr(fn)
is just computed(fn).get()
createTransformer
has been moved to mobx-utils
- Passing
context
explicitly to autorun
, reaction
etc is no longer supported. Use arrow functions or function.bind instead.
- Removed
autorunAsync
. Use the delay
option of autorun
instead.
autorun
, when
, reaction
don't support name as first argument anymore, instead pass the name
option.
- The
extras.
namespace has been dropped to enable tree-shaking non-used MobX features. All methods that where there originally are now exported at top level. If they are part of the official public API (you are encouraged to use them) they are exported as is. If they are experimental or somehow internal (you are discouraged to use them), they are prefixed with _
.
- Dropped bower support. Fixes #1263
- The
spyReportStart
, spyReportEnd
, spyReport
and isSpyEnabled
are no longer public. It is no longer possible to emit custom spy events as to avoid confusing in listeners what the possible set of events is.
- Dropped
isStrictModeEnabled
observable(value)
will only succeed if it can turn the value into an observable data structure (a Map, Array or observable object). But it will no longer create an observable box for other values to avoid confusion. Call observable.box(value)
explictly in such cases.
isComputed
and isObservable
no longer accept a property as second argument. Instead use isComputedProp
and isObservableProp
.
- Removed
whyRun
, use trace
instead
- The spy event signature has slightly changed
- The
Atom
class is no longer exposed. Use createAtom
instead (same signature).
- Calling reportObserved() on a self made atom will no longer trigger the hooks if reportObserved is triggered outside a reactive context.
- The options
struct
and compareStructural
for computed values are deprecated, use `@computed.structor
computed({ equals: comparer.structural})` instead.
isModifierDescriptor
is no longer exposed.
deepEqual
is no longer exposed, use comparer.structural
instead.
setReactionScheduler
-> configure({ reactionScheduler: fn })
reserveArrayBuffer
-> configure({ reactionErrorHandler: fn })
ObservableMap
is no longer exposed as constructor, use observable.map
or isObservableMap
instead
map
-> observable.map
runInAction
no longer accepts a custom scope
- Dropped the already deprecated and broken
default
export that made it harder to tree-shake mobx. Make sure to always use import { x } from "mobx"
and not import mobx from "mobx"
.
- Killed the already deprecated modifiers
asFlat
etc. If you war still using this, see the MobX 2 -> 3 migration notes.
- Observable maps now fully implement the map interface. See #1361 by Marc Fallows
- Observable arrays will no longer expose the
.move
method
- Dropped the
observable.deep.struct
modifier
- Dropped the
observable.ref.struct
modifier
observable.struct
now behaves like observable.ref.struct
(this used to be observable.deep.struct
). That is; values in an observable.struct
field will be stored as is, but structural comparison will be used when assigning a new value
- IReactionDisposer.onError has been removed, use the
onError
option of reactions instead
Issues fixed in this release:
The issues are incoprorated in the above notes.
- #1316 - Improve
observable
api
- #992 -
onBecomeObserved
& onBecomeUnobserved
- #1301 - Set
onError
handler when creating reactions
- #817 - Improve typings of
observe
- #800 - Use
Map
as backend implementation of observable maps
- #1361 - Make observableMaps structurally correct maps
- #813 - Create separate dev and production builds
- #961, #1197 - Make it possible to forbid reading an untracked computed value
- #1098 - Throw instead of warn if multiple MobX instances are active
- #1122 - Atom hooks fired to often for observable maps
- #1148 - Disposer of reactions should also cancel all scheduled effects
- #1241 - Make it possible to disable error boundaries, to make it easier to find exceptions
- #1263 - Remove bower.json
3.6.2
- Fixed accidental dependency on the
node
typings. Fixes #1387 / #1362
3.6.1
3.6.0
- Fixed #1118, the deepEquals operator build into mobx gave wrong results for non-primitive objects. This affected for example
computed.struct
, or the compareStructural
of reaction
3.5.0/1
- Introduced
trace
for easier debugging of reactions / computed values. See the docs for details.
- Improved typings of
observableArray.find
, see #1324 for details.
3.4.1
- Republished 3.4.0, because the package update doesn't seem to distibute consistently through yarn / npm
3.4.0
- Improve Flow support by exposing typings regularly. Flow will automatically include them now. In your
.flowconfig
will have to remove the import in the [libs]
section (as it's done here). Fixes #1232.
3.3.3
- Fixed regression bug where observable map contents could not be replaced using another observable map #1258
- Fixed weird exception abot not being able to read
length
property of a function, see#1238 through #1257 by @dannsam
3.3.2
- Fix bug where custom comparers could be invoked with
undefined
values. Fixes #1208
- Make typings for observable stricter when using flow #1194, #1231
- Fix a bug where
map.replace
would trigger reactions for unchanged values, fixes #1243
- Fixed issue where
NaN
was considered unequal to NaN
when a deep compare was made #1249
3.3.1
- Fix bug allowing maps to be modified outside actions when using strict mode, fixes #940
- Fixed #1139 properly:
transaction
is no longer deprecated and doesn't disable tracking properties anymore
- Fixed #1120:
isComputed
should return false for non-existing properties
3.3.0
- Undeprecated
transaction
, see #1139
- Fixed typings of reaction #1136
- It is now possible to re-define a computed property #1121
- Print an helpful error message when using
@action
on a getter #971
- Improved typings of intercept #1119
- Made code base Prettier #1103
- react-native will now by default use the es module build as well.
- Added support for Weex, see #1163
- Added workaround for Firefox issue causing MobX to crash, see #614
3.2.2
- Fixes a bug (or a known limitation) described in #1092. It is now possible to have different observable administration on different levels of the prototype chain. By @guillaumeleclerc
- Fixed a build issue when using mobx in a project that was using rollup, fixes #1099 by @rossipedia
- Fixed typings of
useStrict
, by @rickbeerendonk
3.2.1
- Introduced customizable value comperators to reactions and computed values.
reaction
and computed
now support an additional option, equals
, which takes a comparision function. See #951 by @jamiewinder. Fixes #802 and #943. See the updated computed
docs for more details.
3.2.0
- MobX will warn again when there are multiple instances of MobX loaded, as this lead to often to confusing bugs if the project setup was not properly. The signal mobx that multiple instances are loaded on purpose, use
mobx.extras.runInSandbox
. See #1082 for details.
3.1.17
- Improved typings of
IObservableArray.intercept
: use more restrictive types for change
parameter of handler
, by @bvanreeven
- Fixed #1072, fields without a default value could not be observed yet when using TypeScript
3.1.16
- Restored
default
export (and added warning), which broke code that was importing mobx like import mobx from "mobx"
. Use import * as mobx from "mobx"
or use named importes instead. By @andykog, see #1043, #1050
- Fixed several typos in exceptions and documentation
3.1.15
- Fixed issue where
array.remove
did not work correctly in combination with extras.interceptReads
3.1.14
- Fixed 3.1.12 / 3.1.13 module packing. See #1039;
module
target is now transpiled to ES5 as well
3.1.13 (Unpublished: Uglify chokes on it in CRA)
- Fixed build issue with webpack 2, see #1040
3.1.12 (Unpublished: wasn't being bundled correctly by all bundlers)
- Added support for ES modules. See #1027 by @rossipedia
- Improved flow typings. See #1019 by @fb55
- Introduced experimental feature
extras.interceptReads(observable: ObservableMap | ObservableArray | ObservableObject | ObservableBox, property?: string, handler: value => value): Disposer
that can be used to intercept reads from observable objects, to transform values on the fly when a value is read. One can achieve similar things with this as with proxying reads. See #1036
3.1.11
- Using rollup as bundler, instead of custom hacked build scripts, by @rossipedia, see #1023
3.1.10
- Fixed flow typings for
when
, by @jamsea
- Add flow typings for
map.replace
, by @leader22
- Added
observableArray.findIndex
, by @leader22
- Improved typings of
autorun
/ autorunAsync
to better support async / await, by @capaj
- Fixed typings of
action.bound
, see #803
3.1.9
- Introduced explicit
.get(index)
and .set(index, value)
methods on observable arrays, for issues that have trouble handling many property descriptors on objects. See also #734
- Made sure it is safe to call
onBecomeObserved
twice in row, fixes #874, #898
- Fixed typings of
IReactionDisposer
3.1.8
- Fixed edge case where
autorun
was not triggered again if a computed value was invalidated by the reaction itself, see #916, by @andykog
- Added support for primtive keys in
createTransformer
, See #920 by @dnakov
- Improved typings of
isArrayLike
, see #904, by @mohsen1
3.1.7
- Reverted ES2015 module changes, as they broke with webpack 2 (will be re-released later)
3.1.6 (Unpublished)
- Expose ES2015 modules to be used with advanced bundlers, by @mohsen1, fixes #868
- Improved typings of
IObservableArray.intercept
: remove superflous type parameter, by @bvanreeven
- Improved typings of map changes, by @hediet
3.1.5
- Improved typings of map changes, see #847, by @hediet
- Fixed issue with
reaction
if fireImmediately
was combined with delay
option, see #837, by @SaboteurSpk
3.1.4
- Observable maps initialized from ES6 didn't deeply convert their values to observables. (fixes #869,by @ggarek)
3.1.3
- Make sure that
ObservableArray.replace
can handle large arrays by not using splats internally. (See e.g. #859)
- Exposed
ObservableArray.spliceWithArray
, that unlike a normal splice, doesn't use a variadic argument list so that it is possible to splice in new arrays that are larger then allowed by the callstack.
3.1.2
3.1.1 (unpublished)
- Introduced
isBoxedObservable(value)
, fixes #804
3.1.0
Improved strict mode
Strict mode has been relaxed a bit in this release. Also computed values can now better handle creating new observables (in an action if needed). The semantics are now as follows:
- In strict mode, it is not allowed to modify state that is already being observed by some reaction.
- It is allowed to create and modify observable values in computed blocks, as long as they are not observed yet.
In order words: Observables that are not in use anywhere yet, are not protected by MobX strict mode.
This is fine as the main goal of strict mode is to avoid kicking of reactions at undesired places.
Also strict mode enforces batched mutations of observables (through action).
However, for unobserved observables this is not relevant; they won't kick of reactions at all.
This fixes some uses cases where one now have to jump through hoops like:
- Creating observables in computed properties was fine already, but threw if this was done with the aid of an action. See issue #798.
- In strict mode, it was not possible to update observable values without wrapping the code in
runInAction
or action
. See issue #563
Note that the following constructions are still anti patterns, although MobX won't throw anymore on them:
- Changing unobserved, but not just created observables in a computed value
- Invoke actions in computed values. Use reactions like
autorun
or reaction
instead.
Note that observables that are not in use by a reaction, but that have .observe
listeners attached, do not count towards being observed.
Observe and intercept callbacks are concepts that do not relate to strict mode, actions or transactions.
Other changes
- Reactions and observable values now consider
NaN === NaN
, See #805 by @andykog
- Merged #783: extract error messages to seperate file, so that they can be optimized in production builds (not yet done), by @reisel, #GoodnessSquad
- Improved typings of actions, see #796 by @mattiamanzati
3.0.2
- Fixed issue where MobX failed on environments where
Map
is not defined, #779 by @dirtyrolf
- MobX can now be compiled on windows as well! #772 by @madarauchiha #GoodnessSquad
- Added documentation on how Flow typings can be used, #766 by @wietsevenema
- Added support for
Symbol.toPrimitive()
and valueOf()
, see #773 by @eladnava #GoodnessSquad
- Supressed an exception that was thrown when using the Chrome Developer tools to inspect arrays, see #752
Re-introduced structural comparison. Seems we couldn't part from it yet :). So the following things have been added:
struct
option to reaction
(alias for compareStructural
, to get more consistency in naming)
observable.struct
, as alias for observable.deep.struct
observable.deep.struct
: Only stores a new value and notify observers if the new value is not structurally the same as the previous value. Beware of cycles! Converts new values automatically to observables (like observable.deep
)
observable.ref.struct
: Only stores a new value and notify observers if the new value is not structurally the same as the previous value. Beware of cycles! Doesn't convert the new value into observables.
extras.deepEquals
: Check if two data structures are deeply equal. supports observable and non observable data structures.
3.0.1
toString()
of observable arrays now behaves like normal arrays (by @capaj, see #759)
- Improved flow types of
toJS
by @jamsea (#758)
3.0.0
The changelog of MobX 3 might look quite overwhelming, but migrating to MobX 3 should be pretty straight forward nonetheless.
The api has now become more layered, and the api is more uniform and modifiers are cleaned up.
In practice, you should check your usage of modifiers (asFlat
, asMap
etc.). Besides that the migration should be pretty painless.
Please report if this isn't the case!
Note that no changes to the runtime algorithm where made, almost all changes evolve in making the creation of observables more uniform, and removing deprecated stuff.
observable
api has been redesigned
The api to create observables has been redesigned.
By default, it keeps the automatic conversion behavior from MobX 2.
However, one can now have more fine grained control on how / which observables are constructed.
Modifiers still exists, but they are more regular, and there should be less need for them.
observable(plainObject)
will no longer enhance objects, but clone instead
When passing a plain object to observable
, MobX used to modify that object inplace and give it observable capabilities.
This also happened when assigning a plain object to an observable array etc.
However, this behavior has changed for a few reasons
- Both arrays and maps create new data structure, however,
observable(object)
didn't
- It resulted in unnecessary and confusing side effects. If you passed an object you received from some api to a function that added it, for example, to an observable collection. Suddenly your object would be modified as side effect of passing it down to that function. This was often confusing for beginners and could lead to subtle bugs.
- If MobX in the future uses Proxies behind the scenes, this would need to change as well
If you want, you can still enhance existing plainObjects, but simply using extendObservable(data, data)
. This was actually the old implementation, which has now changed to extendObservable({}, data)
.
As always, it is best practice not to have transportation objects etc lingering around; there should be only one source of truth, and that is the data that is in your observable state.
If you already adhered to this rule, this change won't impact you.
See #649
Factories per observable type
There are now explicit methods to create an observable of a specific type.
observable.object(props, name?)
creates a new observable object, by cloning the give props and making them observable
observable.array(initialValues, name?)
. Take a guess..
observable.map(initialValues, name?)
observable.box(initialValue, name?)
. Creates a boxed value, which can be read from / written to using .get()
and .set(newValue)
observable(value)
, as-is, based on the type of value
, uses any of the above four functions to create a new observable.
Shallow factories per type
The standard observable factories create observable structures that will try to turn any plain javascript value (arrays, objects or Maps) into observables.
Allthough this is fine in most cases, in some cases you might want to disable this autoconversion.
For example when storing objects from external libraries.
In MobX 2 you needed to use asFlat
or asReference
modifiers for this.
In MobX 3, there are factories to directly create non-converting data structures:
observable.shallowObject(props, name?)
observable.shallowArray(initialValues, name?)
observable.shallowMap(initialValues, name?)
observable.shallowBox(initialValue, name?)
So for example, observable.shallowArray([todo1, todo2])
will create an observable array, but it won't try to convert the todos inside the array into observables as well.
Shallow properties
The @observable
decorator can still be used to introduce observable properties. And like in MobX 2, it will automatically convert its values.
However, sometimes you want to create an observable property that does not convert its value into an observable automatically.
Previously that could be written as @observable x = asReference(value)
.
Structurally comparison of observables have been removed
This was not for a technical reason, but they just seemed hardly used.
Structural comparision for computed properties and reactions is still possible.
Feel free to file an issue, including use case, to re-introduce this feature if you think you really need it.
However, we noticed that in practice people rarely use it. And in cases where it is used reference
/ shallow
is often a better fit (when using immutable data for example).
Modifiers
Modifiers can be used in combination @observable
, extendObservable
and observable.object
to change the autoconversion rules of specific properties.
The following modifiers are available:
observable.deep
: This is the default modifier, used by any observable. It converts any assigned, non-primitive value into an observable value if it isn't one yet.
observable.ref
: Disables automatic observable conversion, just creates an observable reference instead.
observable.shallow
: Can only used in combination with collections. Turns any assigned collection into an collection, which is shallowly observable (instead of deep)
Modifiers can be used as decorator:
class TaskStore {
@observable.shallow tasks = []
}
Or as property modifier in combination with observable.object
/ observable.extendObservable
.
Note that modifiers always 'stick' to the property. So they will remain in effect even if a new value is assigned.
const taskStore = observable({
tasks: observable.shallow([])
})
See modifiers
computed
api has been simplified
Using computed
to create boxed observables has been simplified, and computed
can now be invoked as follows:
computed(expr)
computed(expr, setter)
computed(expr, options)
, where options is an object that can specify one or more of the following fields: name
, setter
, compareStructural
or context
(the "this").
Computed can also be used as a decorator:
@computed
- `@computed.struct
when you want to compareStructural (previously was
@computed({asStructure: true})`)
reaction
api has been simplified
The signature of reaction
is now reaction(dataFunc, effectFunc, options?)
, where the following options are accepted:
context
: The this
to be used in the functions
fireImmediately
delay
: Number in milliseconds that can be used to debounce the effect function.
compareStructural
: false
by default. If true
, the return value of the data function is structurally compared to its previous return value, and the effect function will only be invoked if there is a structural change in the output.
name
: String
Bound actions
It is now possible to create actions and bind them in one go using action.bound
. See #699.
This means that now the following is possible:
class Ticker {
@observable tick = 0
@action.bound
increment() {
this.tick++
}
}
const ticker = new Ticker()
setInterval(ticker.increment, 1000)
Improve error handling
Error handling in MobX has been made more consistent. In MobX 2 there was a best-effort recovery attempt if a derivation throws, but MobX 3 introduced
more consistent behavior:
- Computed values that throw, store the exception and throw it to the next consumer(s). They keep tracking their data, so they are able to recover from exceptions in next re-runs.
- Reactions (like
autorun
, when
, reaction
, render()
of observer
components) will always catch their exceptions, and just log the error. They will keep tracking their data, so they are able to recover in next re-runs.
- The disposer of a reaction exposes an
onError(handler)
method, which makes it possible to attach custom error handling logic to an reaction (that overrides the default logging behavior).
extras.onReactionError(handler)
can be used to register a global onError handler for reactions (will fire after spy "error" event). This can be useful in tests etc.
See #731
Removed error handling, improved error recovery
MobX always printed a warning when an exception was thrown from a computed value, reaction or react component: [mobx] An uncaught exception occurred while calculating....
.
This warning was often confusing for people because they either had the impression that this was a mobx exception, while it actually is just informing about an exception that happened in userland code.
And sometimes, the actual exception was silently caught somewhere else.
MobX now does not print any warnings anymore, and just makes sure its internal state is still stable.
Not throwing or handling an exception is now entirely the responsibility of the user.
Throwing an exception doesn't revert the causing mutation, but it does reset tracking information, which makes it possible to recover from exceptions by changing the state in such a way that a next run of the derivation doesn't throw.
Flow-Types Support 🎉🎉🎉
Flow typings have been added by A-gambit.
Add flow types for methods and interfaces of observable variables:
const observableValue: IObservableValue<number> = observable(1)
const observableArray: IObservableArray<number> = observable([1, 2, 3])
const sum: IComputedValue<number> = computed(() => {
return observableArray.reduce((a: number, b: number): number => a + b, 0)
})
See #640
MobX will no longer share global state by default
For historical reasons (at Mendix), MobX had a feature that it would warn if different versions of the MobX package are being loaded into the same javascript runtime multiple times.
This is because multiple instances by default try to share their state.
This allows reactions from one package to react to observables created by another package,
even when both packages where shipped with their own (embedded) version of MobX (!).
Obviously this is a nasty default as it breaks package isolation and might actually start to throw errors unintentionally when MobX is loaded multiple times in the same runtime by completely unrelated packages.
So this sharing behavior is now by default turned off.
Sharing MobX should be achieved by means of proper bundling, de-duplication of packages or using peer dependencies / externals if needed.
This is similar to packages like React, which will also bail out if you try to load it multiple times.
If you still want to use the old behavior, this can be achieved by running mobx.extras.shareGlobalState()
on all packages that want to share state with each other.
Since this behavior is probably not used outside Mendix, it has been deprecated immediately, so if you rely on this feature, please report in #621, so that it can be undeprecated if there is no more elegant solution.
See #621
Using the @action decorator inside individual objects
Don't use the @action
decorator on an individual object that you pass to observable()
or extendObservable()
. If you have code that looks like observable({ @action f: () => {})
, you should change it to observable({ f: action(() => {})
.
Whether or not this was ever a good idea is debatable, but it stopped working in this version. If you're using classes, it's a non-issue.
Other changes
- Breaking change: The arguments to
observe
listeners for computed and boxed observables have changed and are now consistent with the other apis. Instead of invoking the callback with (newValue: T, oldValue: T)
they are now invoked with a single change object: (change: {newValue: T, oldValue: T, object, type: "update"})
- Using transaction is now deprecated, use
action
or runInAction
instead. Transactions now will enter an untracked
block as well, just as actions, which removes the conceptual difference.
- Upgraded to typescript 2
- It is now possible to pass ES6 Maps to
observable
/ observable maps. The map will be converted to an observable map (if keys are string like)
- Made
action
more debug friendly, it should now be easier to step through
- ObservableMap now has an additional method,
.replace(data)
, which is a combination of clear()
and merge(data)
.
- Passing a function to
observable
will now create a boxed observable refering to that function
- Fixed #603: exceptions in transaction breaks future reactions
- Fixed #698: createTransformer should support default arguments
- Transactions are no longer reported grouped in spy events. If you want to group events, use actions instead.
- Normalized
spy
events further. Computed values and actions now report object
instead of target
for the scope they have been applied to.
- The following deprecated methods have been removed:
transaction
autorunUntil
trackTransitions
fastArray
SimpleEventEmitter
ObservableMap.toJs
(use toJS
)
toJSlegacy
toJSON
(use toJS
)
- invoking
observe
and inject
with plain javascript objects
2.7.0
Automatic inference of computed properties has been deprecated.
A deprecation message will now be printed if creating computed properties while relying on automatical inferrence of argumentless functions as computed values. In other words, when using observable
or extendObservable
in the following manner:
const x = observable({
computedProp: function () {
return someComputation
}
})
x.computedProp
x.computedProp()
Instead, to create a computed property, use:
observable({
get computedProp() {
return someComputation
}
})
or alternatively:
observable({
computedProp: computed(function () {
return someComputation
})
})
This change should avoid confusing experiences when trying to create methods that don't take arguments.
The current behavior will be kept as-is in the MobX 2.* range,
but from MobX 3 onward the argumentless functions will no longer be turned
automatically into computed values; they will be treated the same as function with arguments.
An observable reference to the function will be made and the function itself will be preserved.
See for more details #532
N.B. If you want to introduce actions on an observable that modify its state, using action
is still the recommended approach:
observable({
counter: 0,
increment: action(function () {
this.counter++
})
})
By the way, if you have code such as:
observable({
@computed get someProp() { ... }
});
That code will no longer work. Rather, reactions will fail silently. Remove @computed
.
Note, this only applies when using observable in this way; it doesn't apply when using
@observable
on a property within a class declaration.
Misc
- Fixed #701:
toJS
sometimes failing to convert objects decorated with @observable
(cause: isObservable
sometimes returned false on these object)
- Fixed typings for
when
/ autorun
/ reaction
; they all return a disposer function.
2.6.5
- Added
move
operation to observable array, see #697
2.6.4
- Fixed potential clean up issue if an exception was thrown from an intercept handler
- Improved typings of
asStructure
(by @nidu, see #687)
- Added support for
computed(asStructure(() => expr))
(by @yotambarzilay, see #685)
2.6.3
- Fixed #603: exceptions in transaction breaks future reactions
- Improved typings of
toJS
- Introduced
setReactionScheduler
. Internal api used by mobx-react@4 to be notified when reactions will be run
2.6.2
- Changes related to
toJS
as mentioned in version 2.6.0
where not actually shipped. This has been fixed, so see release notes below.
2.6.1
- Introduced convenience
isArrayLike
: returns whether the argument is either a JS- or observable array. By @dslmeinte
- Improved readme. By @DavidLGoldberg
- Improved assertion message, by @ncammarate (See #618)
- Added HashNode badge, by @sandeeppanda92
2.6.0
Marked as minor release as the behavior of toJS
has been changed, which might be interpreted both as bug-fix or as breaking change, depending of how you interpreted the docs
- Fixed #566: Fixed incorrect behavior of
toJS
: toJS
will now only recurse into observable object, not all objects. The new behavior is now aligned with what is suggested in the docs, but as a result the semantics changed a bit. toJSlegacy
will be around for a while implementing the old behavior. See [#589](See https://github.com/mobxjs/mobx/pull/589) for more details.
- Fixed #571: Don't use
instanceof
operator. Should fix issues if MobX is included multiple times in the same bundle.
- Fixed #576: disallow passing actions directly to
autorun
; as they won't be tracked by @jeffijoe
- Extending observable objects with other observable (objects) is now explicitly forbidden, fixes #540.
2.5.2
- Introduced
isComputed
- Observable objects can now have a type:
IObservableObject
, see #484 by @spiffytech
- Restored 2.4 behavior of boxed observables inside observable objects, see #558
2.5.1
- Computed properties can now be created by using getter / setter functions. This is the idiomatic way to introduce computed properties from now on:
const box = observable({
length: 2,
get squared() {
return this.length * this.length
},
set squared(value) {
this.length = Math.sqrt(value)
}
})
2.5.0
- Core derivation algorithm has received some majore improvements by @asterius1! See below. Pr #452, 489
- Introduced setters for computed properties, use
computed(expr, setter)
or @computed get name() { return expr } set name (value) { action }
. computed
can now be used as modifier in observable
/ extendObservable
, #421, #463 (see below for example)
- Introduced
isStrictModeEnabled()
, deprecated useStrict()
without arguments, see #464
- Fixed #505, accessing an observable property throws before it is initialized
MobX is now able track and memoize computed values while an (trans)action is running.
Before 2.5, accessing a computed value during a transaction always resulted in a recomputation each time the computed value was accessed, because one of the upstream observables (might) have changed.
In 2.5, MobX actively tracks whether one of the observables has changed and won't recompute computed values unnecessary.
This means that computed values are now always memoized for the duration of the current action.
In specific cases, this might signficantly speed up actions that extensively make decisions based on computed values.
Example:
class Square {
@observable length = 2
@computed get squared() {
return this.length * this.length
}
set squared(surfaceSize) {
this.length = Math.sqrt(surfaceSize)
}
@action stuff() {
this.length = 3
console.log(this.squared)
console.log(this.squared)
this.length = 4
console.log(this.squared)
}
}
ES5 example for setters:
function Square() {
extendObservable(this, {
length: 2,
squared: computed(
function () {
return this.squared * this.squared
},
function (surfaceSize) {
this.length = Math.sqrt(surfaceSize)
}
)
})
}
2.4.4
- Fixed #503: map.delete returns boolean
- Fix return type of
runInAction
, #499 by @Strate
- Fixed enumerability of observable array methods, see #496.
- Use TypeScript typeguards, #487 by @Strate
- Added overloads to
action
for better type inference, #500 by @Strate
- Fixed #502:
extendObservable
fails on objects created with Object.create(null)
- Implemented #480 / #488: better typings for
asMap
, by @Strate
2.4.3
- Objects with a
null
prototype are now considered plain objects as well
- Improved error message for non-converging cyclic reactions
- Fixed potential HMR issue
2.4.2
- Improved error message when wrongly using
@computed
, by @bb (#450)
observableArray.slice
now automatically converts observable arrays to plain arrays, fixes #460
- Improved error message when an uncaught exception is thrown by a MobX tracked function
2.4.1
@action
decorated methods are now configurable. Fixes #441
- The
onBecomeObserved
event handler is now triggered when an atom is observed, instead of when it is bound as dependency. Fixes #427 and makes atoms easier to extend.
- if
useStrict()
is invoked without arguments, it now returns the current value of strict mode.
- the current reaction is now always passed as first argument to the callbacks of
autorun
, autorunAsync
, when
and reaction
. This allows reactions to be immediately disposed during the first run. See #438, by @andykog
2.4.0
- Note: the internal version of MobX has been bumped. This version has no breaking api changes, but if you have MobX loaded multiple times in your project, they all have to be upgraded to
2.4.0
. MobX will report this when starting.
- Made dependency tracking and binding significant faster. Should result in huge performance improvements when working with large collections.
- Fixed typescript decorator issue, #423, #425? (by @bb)
2.3.7
- Fixed issue where computed values were tracked and accidentally kept alive during actions
2.3.6
- Fixed #406: Observable maps doesn't work with empty initial value in Safari
- Implemented #357, #348: ObservableMap and ObservableArray now support iterators. Use
@@iterator()
or iterall in ES5 environments.
2.3.5
- Fixed #364: Observable arrays not reacting properly to index assignments under iOS safari (mobile) 9.1.1 By @andykog
- Fixed #387: Typings of boxed values
- Added warning when reading array entries out of bounds. See #381
2.3.4
- Fixed #360: Removed expensive cycle detection (cycles are still detected, but a bit later)
- Fixed #377:
toJS
serialization of Dates and Regexes preserves the original values
- Fixed #379:
@action
decorated methods can now be inherited / overridden
2.3.3
- Fixed #186: Log a warning instead of an error if an exception is thrown in a derivation. Fixes issue where React Native would produce unusable error screens (because it shows the first logged error)
- Fixed #333: Fixed some interoperability issues in combination with
Reflect
/ InversifyJS
decorators. @andykog
- Fixed #333:
@observable
class properties are now owned by their instance again, meaning they will show up in Object.keys()
and .hasOwnProperty
@andykog
2.3.2
- Fixed #328: Fixed exception when inspecting observable in
onBecomeObserved
- Fixed #341:
array.find
now returns undefined
instead of null
when nothing was found, behavior now matches the docs. (By @hellectronic)
2.3.1
- Fixed #327: spy not working with runInAction
2.3.0
Introduced whyRun
:
Usage:
whyRun()
whyRun(Reaction object / ComputedValue object / disposer function)
whyRun(object, "computed property name")
whyRun
is a small utility that can be used inside computed value or reaction (autorun
, reaction
or the render
method of an observer
React component)
and prints why the derivation is currently running, and under which circumstances it will run again.
This should help to get a deeper understanding when and why MobX runs stuff, and prevent some beginner mistakes.
This feature can probably be improved based on your feedback, so feel free to file issues with suggestions!
Semantic changes:
@observable
is now always defined on the class prototypes and not in the instances. This means that @observable
properties are enumerable, but won't appear if Object.keys
or hasOwnProperty
is used on a class instance.
- Updated semantics of
reaction
as discussed in #278
. The expression now needs to return a value and the side effect won't be triggered if the result didn't change. asStructure
is supported in these cases. In contrast to MobX 2.2, effects will no longer be run if the output of the expression didn't change.
Enhancements
- Introduces
isAction(fn)
#290
- If an (argumentless) action is passed to
observable
/ extendObservable
, it will not be converted into a computed property.
- Fixed #285: class instances are now also supported by
toJS
. Also members defined on prototypes which are enumerable are converted.
- Map keys are now always coerced to strings. Fixes #308
when
, autorun
and autorunAsync
now accept custom debug names (see #293, by @jamiewinder)
- Fixed #286: autoruns no longer stop working if an action throws an exception
- Implemented
runInAction
, can be used to create on the fly actions (especially useful in combination with async/await
, see #299
- Improved performance and reduced mem usage of decorators signficantly (by defining the properties on the prototype if possible), and removed subtle differences between the implementation and behavior in babel and typescript.
- Updated logo as per #244. Tnx @osenvosem!
2.2.2:
- Fixed issue #267: exception when
useStrict(true)
was invoked in combination with @observable
attributes when using Babel
- Fixed issue #269: @action in combination with typescript targeting ES6 and reflect.ts
- Improved compatibility with
JSON.stringify
, removed incorrect deprecation message
- Improved some error messages
2.2.1
- Fixed issue where typescript threw a compile error when using
@action
without params on a field
- Fixed issue where context was accidentally shared between class instances when using
@action
on a field
2.2.0
See the release announcement for the full details of this release:
Introduced:
action
/ @action
intercept
spy
reaction
useStrict
- improved debug names
toJSON
was renamed to toJS
observable(asMap())
is the new idiomatic way to create maps
- the effect of
when
is now untracked, similar to `reaction.
extras.trackTransations
is deprecated, use spy
instead
untracked
has been undeprecated
- introduced / documented:
getAtom
, getDebugName
, isSpyEnabled
, spyReport
, spyReportStart
, spyReportEnd
- deprecated
extras.SimpleEventEmitter
- array splice events now also report the
added
collection and removedCount
2.1.7
- Fixed a false negative in cycle detection, as reported in #236
2.1.6
- Fixed #236, #237 call stack issues when working with large arrays
2.1.5
- Fix #222 (by @andykog) run
observe
callback of computed properties in untracked mode.
2.1.4
- Fixed #201 (see also #160), another iOS enumerability issue... By @luosong
2.1.3
- Fixed #191, when using babel, complex field initializers where shared. By @andykog
- Added
lib/mobx.umd.min.js
for minified cdn builds, see #85
2.1.2
- Improved debug names of objects created using a constructor
- Fixed(?) some issues with iOS7 as reported in #60 by @bstst
2.1.1
- Fixed issue where
autorun
s created inside autorun
s were not always kicked off. (mobx-react
's observer
was not affected). Please upgrade if you often use autorun.
- Fixed typings of
mobx.map
, a list of entries is also acceptable.
- (Experimental) Improved error recovery a bit further
2.1.0
- MobX is now chatty again when an exception occurs inside a autorun / computed value / React.render. Previously this was considered to be the responsibility of the surrounding code. But if exceptions were eaten this would be really tricky to debug.
- (Experimental) MobX will now do a poor attempt to recover from exceptions that occured in autorun / computed value / React.render.
2.0.6
resetGlobalState
is now part of the mobx.extras
namespace, as it is useful for test setup, to restore inconsistent state after test failures.
resetGlobalState
now also resets the caches of createTransformer
, see #163.
2.0.5
2.0.4
$transformId
property on transformed objects should be non-enumerable. Fixes #170.
2.0.3
- Always peek if inspecting a stale, computed value. Fixes #165.
2.0.2
- Fixed issue where changing an object property was tracked, which could lead to unending loops in
autorunAsync
.
2.0.1
- Undeprecated
observable(scalar)
(see 143)
expr
no longer prints incorrect deprecated messages (see 143)
- Requires
mobx
twice no longer fails.
2.0.0
A new name...
Welcome to ~Mobservable~ MobX 2! First of all, there is the name change.
The new name is shorter and funnier and it has the right emphasis: MobX is about reactive programming.
Not about observability of data structures, which is just a technical necessity.
MobX now has its own mobxjs organization on GitHub. Just report an issue if you want to join.
All MobX 2.0 two compatible packages and repos have been renamed. So mobx-react
, mobx-react-devtools
etc.
For the 1.0 versions, use the old mobservable
based names.
Migrating from Mobservable 1.x to MobX 2.0
Migrating from Mobservable should be pretty straight-forward as the public api is largely the same.
However there are some conceptual changes which justifies a Major version bump as it might alter the behavior of MobX in edge cases.
Besides that, MobX is just a large collection of minor improvements over Mobservable.
Make sure to remove your old mobservable
dependencies when installing the new mobx
dependencies!
autorun
s are now allowed to cause cycles!
autorun
is now allowed to have cycles. In Mobservable 1 an exception was thrown as soon as an autorun modified a variable which it was reading as well.
In MobX 2 these situations are now allowed and the autorun will trigger itself to be fired again immediately after the current execution.
This is fine as long as the autorun terminates within a reasonable amount of iterations (100).
This should avoid the need for work-arounds involving setTimeout
etc.
Note that computed values (created using observable(func)
are still not allowed to have cycles.
[Breaking] observable(scalar)
returns an object instead of a function and has been deprecated.
Creating an observable from a primitive or a reference no longer returns a getter/setter function, but a method with a .get
and .set
method.
This is less confusing, easier to debug and more efficient.
So to read or write from an observable scalar use:
const temperature = observable(27)
temperature.set(15)
temperature.get()
observable(scalar)
has been deprecated to make the api smaller and the syntax more uniform. In practice having observable objects, arrays and decorators seems to suffice in 99% of the cases. Deprecating this functionality means that people have simply less concepts to learn. Probably creating observable scalars will continue to work for a long time, as it is important to the internals of MobX and very convenient for testing.
Introduced @computed
MobX introduced the @computed
decorator for ES6 class properties with getter functions.
It does technically the same as @observable
for getter properties. But having a separate decorator makes it easier to communicate about the code.
@observable
is for mutable state properties, @computed
is for derived values.
@computed
can now also be parameterized. @computed({asStructure: true})
makes sure that the result of a derivation is compared structurally instead of referentially with its preview value. This makes sure that observers of the computation don't re-evaluate if new structures are returned that are structurally equal to the original ones. This is very useful when working with point, vector or color structures for example. It behaves the same as the asStructure
modifier for observable values.
@computed
properties are no longer enumerable.
MobX is now extensible!
The core algorithm of MobX has been largely rewritten to improve the clarity, extensibility, performance and stability of the source code.
It is now possible to define your own custom observable data sources by using the Atom
class.
It is also possible to create your own reactive functions using the Reaction
class. autorun
, autorunAsync
and @observer
have now all been implemented using the concept of Reactions.
So feel free to write your own reactive constructions!
Mobservable now fails fast
In Mobservable 1 exceptions would be caught and sometimes re-thrown after logging them.
This was confusing and not all derivations were able to recover from these exceptions.
In MobX 2 it is no longer allowed for a computed function or autorun
to throw an exception.
Improved build
Other changes
- Improved debug names of all observables. This is especially visible when using
mobx-react-devtools
or extras.trackTransitions
.
- Renamed
extras.SimpleEventEmitter
to SimpleEventEmitter
- Removed already deprecated methods:
isReactive
, makeReactive
, observeUntil
, observeAsync
- Removed
extras.getDNode
- Invoking
ObservableArray.peek
is no longer registered as listener
- Deprecated
untracked
. It wasn't documented and nobody seems to miss it.
1.2.5
- Map no longer throws when
.has
, .get
or .delete
is invoked with an invalid key (#116)
- Files are now compiled without sourcemap to avoid issues when loading mobservable in a debugger when
src/
folder is not available.
1.2.4
- Fixed: observable arrays didn't properly apply modifiers if created using
asFlat([])
or fastArray([])
- Don't try to make frozen objects observable (by @andykog)
observableArray.reverse
no longer mutates the arry but just returns a sorted copy
- Updated tests to use babel6
1.2.3
- observableArray.sort no longer mutates the array being sorted but returns a sorted clone instead (#90)
- removed an incorrect internal state assumption (#97)
1.2.2
1.2.1
- Computed value now yields consistent results when being inspected while in transaction
1.2.0
1.1.8
- Implemented #59,
isObservable
and observe
now support a property name as second param to observe individual values on maps and objects.
1.1.7
- Fixed #77: package consumers with --noImplicitAny should be able to build
1.1.6
- Introduced
mobservable.fastArray(array)
, in addition to mobservable.observable(array)
. Which is much faster when adding items but doesn't support enumerability (for (var idx in ar) ..
loops).
- Introduced
observableArray.peek()
, for fast access to the array values. Should be used read-only.
1.1.5
- Fixed 71: transactions should not influence running computations
1.1.4
- Fixed #65; illegal state exception when using a transaction inside a reactive function. Credits: @kmalakoff
1.1.3
- Fixed #61; if autorun was created during a transaction, postpone execution until the end of the transaction
1.1.2
- Fixed exception when autorunUntil finished immediately
1.1.1
toJSON
now serializes object trees with cycles as well. If you know the object tree is acyclic, pass in false
as second parameter for a performance gain.
1.1.0
- Exposed
ObservableMap
type
- Introduced
mobservable.untracked(block)
- Introduced
mobservable.autorunAsync(block, delay)
1.0.9
Removed accidental log message
1.0.7 / 1.0.8
Fixed inconsistency when using transaction
and @observer
, which sometimes caused stale values to be displayed.
1.0.6
Fix incompatibility issue with systemjs bundler (see PR 52)
1.0.4/5
map.size
is now a property instead of a function
map()
now accepts an array as entries to construct the new map
- introduced
isObservableObject
, isObservableArray
and isObservableMap
- introduced
observe
, to observe observable arrays, objects and maps, similarly to Object.observe and Array.observe
1.0.3
extendObservable
now supports passing in multiple object properties
1.0.2
- added
mobservable.map()
, which creates a new map similarly to ES6 maps, yet observable. Until properly documentation, see the MDN docs.
1.0.1
- Stricter argument checking for several apis.
1.0
Renames
isReactive
-> isObservable
makeReactive
-> observable
extendReactive
-> extendObservable
observe
-> autorun
observeUntil
-> autorunUntil
observeAsync
-> autorunAsync
reactiveComponent
-> observer
(in mobservable-react
package)
Breaking changes
- dropped the
strict
and logLevel
settings of mobservable. View functions are by default run in strict
mode, autorun
(formerly: observe
) functions in non-strict
mode (strict indicates that it is allowed to change other observable values during the computation of a view funtion).
Use extras.withStrict(boolean, block)
if you want to deviate from the default behavior.
observable
(formerly makeReactive
) no longer accepts an options object. The modifiers asReference
, asStructure
and asFlat
can be used instead.
- dropped the
default
export of observable
- Removed all earlier deprecated functions
Bugfixes / improvements
mobservable
now ships with TypeScript 1.6 compliant module typings, no external typings file is required anymore.
mobservable-react
supports React Native as well through the import "mobservable-react/native"
.
- Improved debugger support
for (var key in observablearray)
now lists the correct keys
@observable
now works correct on classes that are transpiled by either TypeScript or Babel (Not all constructions where supported in Babel earlier)
- Simplified error handling, mobservable will no longer catch errors in views, which makes the stack traces easier to debug.
- Removed the initial 'welcom to mobservable' logline that was printed during start-up.
0.7.1
- Backported Babel support for the @observable decorator from the 1.0 branch. The decorator should now behave the same when compiled with either Typescript or Babeljs.
0.7.0
- Introduced
strict
mode (see issues #30, #31)
- Renamed
sideEffect
to observe
- Renamed
when
to observeUntil
- Introduced
observeAsync
.
- Fixed issue where changing the
logLevel
was not picked up.
- Improved typings.
- Introduces
asStructure
(see #8) and asFlat
.
- Assigning a plain object to a reactive structure no longer clones the object, instead, the original object is decorated. (Arrays are still cloned due to Javascript limitations to extend arrays).
- Reintroduced
expr(func)
as shorthand for makeReactive(func)()
, which is useful to create temporarily views inside views
- Deprecated the options object that could be passed to
makeReactive
.
- Deprecated the options object that could be passed to
makeReactive
:
- A
thisArg
can be passed as second param.
- A name (for debugging) can be passed as second or third param
- The
as
modifier is no longer needed, use asReference
(instead of as:'reference'
) or asFlat
(instead of recurse:false
).
0.6.10
- Fixed issue where @observable did not properly create a stand-alone view
0.6.9
- Fixed bug where views where sometimes not triggered again if the dependency tree changed to much.
0.6.8
- Introduced
when
, which, given a reactive predicate, observes it until it returns true.
- Renamed
sideEffect -> observe
0.6.7:
0.6.6:
- Deprecated observable array
.values()
and .clone()
- Deprecated observeUntilInvalid; use sideEffect instead
- Renamed mobservable.toJson to mobservable.toJSON
0.6.5:
- It is no longer possible to create impure views; views that alter other reactive values.
- Update links to the new documentation.
0.6.4:
- 2nd argument of sideEffect is now the scope, instead of an options object which hadn't any useful properties
0.6.3
- Deprecated: reactiveComponent, reactiveComponent from the separate package mobservable-react should be used instead
- Store the trackingstack globally, so that multiple instances of mobservable can run together
0.6.2
- Deprecated: @observable on functions (use getter functions instead)
- Introduced:
getDependencyTree
, getObserverTree
and trackTransitions
- Minor performance improvements