Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: vuejs/composition-api
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: v0.3.2
Choose a base ref
...
head repository: vuejs/composition-api
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: v0.3.3
Choose a head ref
  • 9 commits
  • 13 files changed
  • 7 contributors

Commits on Sep 19, 2019

  1. chore: setup CI

    liximomo authored Sep 19, 2019

    Verified

    This commit was created on GitHub.com and signed with GitHub’s verified signature.
    Copy the full SHA
    db81a77 View commit details

Commits on Sep 20, 2019

  1. fix(src/install.ts): fix typo

    a20185 authored and liximomo committed Sep 20, 2019
    Copy the full SHA
    bacf216 View commit details

Commits on Sep 28, 2019

  1. Copy the full SHA
    bb1e030 View commit details

Commits on Oct 31, 2019

  1. fix: make __ob__ unenumerable (#149)

    * fix: make __ob__ unenumerable
    pigwang authored and liximomo committed Oct 31, 2019
    Copy the full SHA
    1672b6e View commit details

Commits on Nov 22, 2019

  1. fix: fix computed type

    * fix unwritable computed type
    
    * Update src/apis/computed.ts
    
    Co-Authored-By: Carlos Rodrigues <david-181@hotmail.com>
    
    * Prevent params from being changed
    
    * [fix] fix for review
    
    * Revert "[fix] fix for review"
    
    This reverts commit d6a6eac.
    kahirokunn authored and liximomo committed Nov 22, 2019
    Copy the full SHA
    b9d8a4d View commit details

Commits on Dec 1, 2019

  1. feat: expose getCurrentInstance like in Vue 3

    Guillaume Chau committed Dec 1, 2019
    Copy the full SHA
    6019f67 View commit details
  2. chore: fix comment typo

    Guillaume Chau committed Dec 1, 2019
    Copy the full SHA
    b0a5c34 View commit details

Commits on Dec 2, 2019

  1. feat: add onServerPrefetch (#198)

    * feat: add `onServerPrefetch`
    
    * test: onServerPrefetch
    
    * docs: note about SSR
    
    * test: ssrContext
    
    * feat: ssrContext in setup context
    
    * test: improved shared context test
    Akryum authored Dec 2, 2019
    Copy the full SHA
    7456343 View commit details
  2. build: release 0.3.3

    Guillaume Chau committed Dec 2, 2019
    Copy the full SHA
    cbfb18d View commit details
Showing with 306 additions and 8 deletions.
  1. +29 −0 .github/workflows/nodejs.yml
  2. +27 −0 README.md
  3. +5 −0 README.zh-CN.md
  4. +3 −2 package.json
  5. +5 −3 src/apis/computed.ts
  6. +1 −0 src/apis/lifecycle.ts
  7. +3 −0 src/index.ts
  8. +1 −1 src/install.ts
  9. +1 −1 src/reactivity/reactive.ts
  10. +8 −1 src/setup.ts
  11. +18 −0 test/setup.spec.js
  12. +139 −0 test/ssr/serverPrefetch.spec.js
  13. +66 −0 yarn.lock
29 changes: 29 additions & 0 deletions .github/workflows/nodejs.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
name: Node CI

on:
pull_request:
branches:
- master

jobs:
build:

runs-on: ubuntu-latest

strategy:
matrix:
node-version: [8.x, 10.x, 12.x]

steps:
- uses: actions/checkout@v1
- name: Use Node.js ${{ matrix.node-version }}
uses: actions/setup-node@v1
with:
node-version: ${{ matrix.node-version }}
- name: npm install, build, and test
run: |
npm install
npm run build --if-present
npm test
env:
CI: true
27 changes: 27 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -167,6 +167,11 @@ b.list.push(
b.list[1].count === 1; // true
```

### ***Using*** `reactive` will mutate the origin object

This is an limitation of using `Vue.observable` in Vue 2.
> Vue 3 will return an new proxy object.
---

## `watch()` API
@@ -306,3 +311,25 @@ declare module '@vue/composition-api/dist/component/component' {
}
}
```

## SSR

Even if there is no definitive Vue 3 API for SSR yet, this plugin implements the `onServerPrefetch` lifecycle hook that allows you to use the `serverPrefetch` hook found in the classic API.

```js
import { onServerPrefetch } from '@vue/composition-api';

export default {
setup (props, { ssrContext }) {
const result = ref();

onServerPrefetch(async () => {
result.value = await callApi(ssrContext.someId);
});

return {
result,
};
},
};
```
5 changes: 5 additions & 0 deletions README.zh-CN.md
Original file line number Diff line number Diff line change
@@ -167,6 +167,11 @@ b.list.push(
b.list[1].count === 1; // true
```

### `reactive` 会返回一个修改过的原始的对象

此行为与 Vue 2 中的 `Vue.observable` 一致
> Vue 3 中会返回一个新的的代理对象.
---

## `watch()` API
5 changes: 3 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@vue/composition-api",
"version": "0.3.2",
"version": "0.3.3",
"description": "Provide logic composition capabilities for Vue.",
"keywords": [
"vue",
@@ -46,7 +46,8 @@
"ts-jest": "^24.0.2",
"typescript": "^3.6.2",
"vue": "^2.5.22",
"vue-router": "^3.1.3"
"vue-router": "^3.1.3",
"vue-server-renderer": "^2.6.10"
},
"peerDependencies": {
"vue": "^2.5.22"
8 changes: 5 additions & 3 deletions src/apis/computed.ts
Original file line number Diff line number Diff line change
@@ -9,11 +9,13 @@ interface Option<T> {
}

// read-only
export function computed<T>(getter: Option<T>['get']): Ref<T>;
export function computed<T>(getter: Option<T>['get']): Readonly<Ref<Readonly<T>>>;
// writable
export function computed<T>(options: Option<T>): Ref<T>;
export function computed<T>(options: Option<T>): Ref<Readonly<T>>;
// implement
export function computed<T>(options: Option<T>['get'] | Option<T>): Ref<T> {
export function computed<T>(
options: Option<T>['get'] | Option<T>
): Readonly<Ref<Readonly<T>>> | Ref<Readonly<T>> {
const vm = getCurrentVM();
let get: Option<T>['get'], set: Option<T>['set'] | undefined;
if (typeof options === 'function') {
1 change: 1 addition & 0 deletions src/apis/lifecycle.ts
Original file line number Diff line number Diff line change
@@ -38,3 +38,4 @@ export const onUnmounted = createLifeCycles(['destroyed', 'deactivated'], genNam
export const onErrorCaptured = createLifeCycle('errorCaptured');
export const onActivated = createLifeCycle('activated');
export const onDeactivated = createLifeCycle('deactivated');
export const onServerPrefetch = createLifeCycle('serverPrefetch');
3 changes: 3 additions & 0 deletions src/index.ts
Original file line number Diff line number Diff line change
@@ -24,6 +24,9 @@ export default plugin;
export { default as createElement } from './createElement';
export { SetupContext };
export { createComponent, ComponentRenderProxy, PropType, PropOptions } from './component';
// For getting a hold of the interal instance in setup() - useful for advanced
// plugins
export { getCurrentVM as getCurrentInstance } from './runtimeContext';

export * from './apis/state';
export * from './apis/lifecycle';
2 changes: 1 addition & 1 deletion src/install.ts
Original file line number Diff line number Diff line change
@@ -26,7 +26,7 @@ function mergeData(to: AnyObject, from?: AnyObject): Object {
} else if (
toVal !== fromVal &&
(isPlainObject(toVal) && !isRef(toVal)) &&
(isPlainObject(fromVal) && !isRef(toVal))
(isPlainObject(fromVal) && !isRef(fromVal))
) {
mergeData(toVal, fromVal);
}
2 changes: 1 addition & 1 deletion src/reactivity/reactive.ts
Original file line number Diff line number Diff line change
@@ -152,7 +152,7 @@ export function nonReactive<T = any>(obj: T): T {
}

// set the vue observable flag at obj
(obj as any).__ob__ = (observe({}) as any).__ob__;
def(obj, '__ob__', (observe({}) as any).__ob__);
// mark as nonReactive
def(obj, NonReactiveIdentifierKey, NonReactiveIdentifier);

9 changes: 8 additions & 1 deletion src/setup.ts
Original file line number Diff line number Diff line change
@@ -216,7 +216,14 @@ export function mixin(Vue: VueConstructor) {
const ctx = {
slots: {},
} as SetupContext;
const props: Array<string | [string, string]> = ['root', 'parent', 'refs', 'attrs', 'listeners'];
const props: Array<string | [string, string]> = [
'root',
'parent',
'refs',
'attrs',
'listeners',
'ssrContext',
];
const methodReturnVoid = ['emit'];
props.forEach(key => {
let targetKey: string;
18 changes: 18 additions & 0 deletions test/setup.spec.js
Original file line number Diff line number Diff line change
@@ -276,6 +276,24 @@ describe('setup', () => {
.then(done);
});

it("should put a unenumerable '__ob__' for non-reactive object", () => {
const clone = obj => JSON.parse(JSON.stringify(obj));
const componentSetup = jest.fn(props => {
const internalOptions = clone(props.options);
return { internalOptions };
});
const ExternalComponent = {
props: ['options'],
setup: componentSetup,
};
new Vue({
components: { ExternalComponent },
setup: () => ({ options: {} }),
template: `<external-component :options="options"></external-component>`,
}).$mount();
expect(componentSetup).toReturn();
});

it('current vue should exist in nested setup call', () => {
const spy = jest.fn();
new Vue({
139 changes: 139 additions & 0 deletions test/ssr/serverPrefetch.spec.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,139 @@
const Vue = require('vue/dist/vue.common.js');
const { createRenderer } = require('vue-server-renderer');
const { ref, onServerPrefetch, getCurrentInstance, provide, inject } = require('../../src');

function fetch(result) {
return new Promise(resolve => {
setTimeout(() => {
resolve(result);
}, 10);
});
}

describe('serverPrefetch', () => {
it('should prefetch async operations before rendering', async () => {
const app = new Vue({
setup() {
const count = ref(0);

onServerPrefetch(async () => {
count.value = await fetch(42);
});

return {
count,
};
},
render(h) {
return h('div', this.count);
},
});

const serverRenderer = createRenderer();
const html = await serverRenderer.renderToString(app);
expect(html).toBe('<div data-server-rendered="true">42</div>');
});

it('should prefetch many async operations before rendering', async () => {
const app = new Vue({
setup() {
const count = ref(0);
const label = ref('');

onServerPrefetch(async () => {
count.value = await fetch(42);
});

onServerPrefetch(async () => {
label.value = await fetch('meow');
});

return {
count,
label,
};
},
render(h) {
return h('div', [this.count, this.label]);
},
});

const serverRenderer = createRenderer();
const html = await serverRenderer.renderToString(app);
expect(html).toBe('<div data-server-rendered="true">42meow</div>');
});

it('should pass ssrContext', async () => {
const child = {
setup(props, { ssrContext }) {
const content = ref();

expect(ssrContext.foo).toBe('bar');

onServerPrefetch(async () => {
content.value = await fetch(ssrContext.foo);
});

return {
content,
};
},
render(h) {
return h('div', this.content);
},
};

const app = new Vue({
components: {
child,
},
render(h) {
return h('child');
},
});

const serverRenderer = createRenderer();
const html = await serverRenderer.renderToString(app, { foo: 'bar' });
expect(html).toBe('<div data-server-rendered="true">bar</div>');
});

it('should not share context', async () => {
const instances = [];
function createApp(context) {
return new Vue({
setup() {
const count = ref(0);

onServerPrefetch(async () => {
count.value = await fetch(context.result);
});

instances.push(getCurrentInstance());

return {
count,
};
},
render(h) {
return h('div', this.count);
},
});
}

const serverRenderer = createRenderer();
const promises = [];
// Parallel requests
for (let i = 1; i < 3; i++) {
promises.push(
new Promise(async resolve => {
const app = createApp({ result: i });
const html = await serverRenderer.renderToString(app);
expect(html).toBe(`<div data-server-rendered="true">${i}</div>`);
resolve();
})
);
}
await Promise.all(promises);
expect((instances[0] === instances[1]) === instances[2]).toBe(false);
});
});
Loading