Karena kenyataan bahwa Bereaksi menyediakan kemampuan luar biasa untuk bekerja dengan tampilan, Anda hanya dapat fokus pada pengorganisasian logika aplikasi dan semantik kode yang menjelaskan cara bekerja dengan data. Itu memilih perpustakaan manajemen negara, pilihan gaya basis kode masa depan terjadi.
Dalam artikel ini, kami akan mempertimbangkan pendekatan berbasis layanan toko diimplementasikan melalui perpustakaan lamp-luwak
.

TL; DR Siapa pun yang ingin segera mulai menulis kode dapat langsung beralih ke paragraf berikutnya.
Semua logika di dalam aplikasi Anda dibagi ke dalam layanan, setiap layanan berisi satu sisi, perubahan yang memberi tahu pelanggan. Layanan ini juga berisi fungsi logika di mana Anda dapat mengakses layanan lain dan menulis ulang sisi yang tidak dapat diubah, fungsi-fungsi ini dapat tidak sinkron dan mengandung efek samping.
lamp-luwak
, . , React useProvide
useSubscribe
. , , .
, - . , embedded . .
subscribe
. .
, todo- :
todos create-react-app
npx create-react-app todos --template typescript --use-npm
yarn create react-app todos --template typescript
lamp-luwak
todos
npm i --save lamp-luwak
yarn add lamp-luwak
, React - , - .
— -, , .
-, store
.
class Todos {
store = [ ];
store
. .
class TodoCounters {
todo = provide(Todo);
provide
, , Todo
.
React
const List = () => {
const todo = useProvide(Todo);
};
-. React useProvide
, — - , Todo
. , .
— , , store
provide
useProvide
/ , .
, List Todo
.
services
, React components
.
sr/
components/ // React
Counters.tsx //
Input.tsx //
Task.tsx //
List.tsx //
services/ //
Todo/
Task.ts //
Todo.ts //
TodoCounters.ts //
App.tsx //
2- . :
Todo
— , add
;TodoCounters
— .
Todo
, Task
, .. , , . , , . , Date, Map, Set. ( SSR) , UI. , . , Task
toggle
, .
Task
, , create
, -, - .
import { create } from 'lamp-luwak';
import { Task } from './Todo/Task';
export class Todo {
store = [
create(Task, { id: 1, label: 'Cook the dinner', completed: false }),
create(Task, { id: 2, label: 'Cook the breakfast', completed: true })
]
add(label: string) {
this.store = this.store.concat(
create(Task, { id: Date.now(), label, completed: false })
);
}
}
Todo
. store
Task
, create
. add
, , Task
. id
, Date.now
.
import { subscribe, modify, action } from 'lamp-luwak';
type Store = {
id: number,
label: string,
completed: boolean
}
export const TaskChanged = action();
export class Task {
store: Store;
constructor(store: Store) {
this.store = store;
subscribe(this, TaskChanged);
}
toggle() {
modify(this).completed = !this.store.completed;
}
}
Task
, . : id
, label
completed
. toggle
, , modify
, , .
TaskChanged
, action
.
— , . , — dispatch
lamp-luwak
.
, , TaskChanged
Task
. subscribe
, — , .. , — , . TaskChanged
, .
import { provide, subscribe } from 'lamp-luwak';
import { Todo } from './Todo';
import { TaskChanged } from './Todo/Task';
export class TodoCounters {
todo = provide(Todo);
store = {
active: 0,
completed: 0
}
constructor() {
subscribe(this.todo, this.calculate, this);
subscribe(TaskChanged, this.calculate, this);
this.calculate();
}
calculate() {
const items = this.todo.store;
const completed = items.filter(item => item.store.completed).length;
const active = items.length - completed;
this.store = { completed, active };
}
}
TodoCounters
, . 2- :
Todo
, , .TaskChanged
, - Task
, completed
, .
calculate
.
:

App.tsx
, .
List
— .
import React from 'react';
import { useProvide } from 'lamp-luwak';
import { Todo } from '../services/Todo';
import { Task } from './Task';
export const List = () => {
const todo = useProvide(Todo);
const items = todo.store;
if (items.length === 0) return null;
return (
<ul>
{items.map(item => (
<Task task={item} key={item.store.id} />
))}
</ul>
)
};
Todo
, useProvide
, List
, , , . , .
Task
— .
import React, { FC } from 'react';
import { useSubscribe } from 'lamp-luwak';
import { Task as TaskClass } from '../services/Todo/Task';
export const Task: FC<{ task: TaskClass }> = ({ task }) => {
useSubscribe(task);
const { label, completed } = task.store;
return (
<li>
<input
className="toggle"
type="checkbox"
checked={completed}
onChange={() => task.toggle()}
/>
<span style={{
textDecoration: completed ? 'line-through' : 'none'
}}>
{label}
</span>
</li>
)
};
Task
, , useSubscribe
, . , toggle
.
Counters
— .
import React from 'react';
import { useProvide } from 'lamp-luwak';
import { TodoCounters } from '../services/TodoCounters';
export const Counters = () => {
const { active, completed } = useProvide(TodoCounters).store;
return (
<>
<div>Active: {active}</div>
<div>Completed: {completed}</div>
</>
)
};
, useProvide
. , TodoCounters
.
Input
— .
add
Todo
. , Todo
, React useState
.
import React, { useState } from 'react';
import { useProvide } from 'lamp-luwak';
import { Todo } from '../services/Todo';
export const Input = () => {
const [text, setText] = useState('Cook the lunch');
const todo = useProvide(Todo);
const add = () => {
todo.add(text);
setText('');
};
return (
<>
<input
onChange={(e) => setText(e.target.value)}
value={text}
autoFocus
onKeyDown={(event: any) => {
if (event.keyCode === 13) add();
}}
/>
<button onClick={add}>Add</button>
</>
);
};
App.tsx
, src
, .
import React from 'react';
import { Input } from './components/Input';
import { List } from './components/List';
import { Counters } from './components/Counters';
const App = () => (
<>
<Input />
<List />
<Counters />
</>
);
export default App;
Enjoy! .
lamp-luwak
, , . .
, .
, , , .
.