ホームページ ウェブフロントエンド jsチュートリアル Immutable.js で元に戻すおよびやり直し機能を実装する方法 (詳細なチュートリアル)

Immutable.js で元に戻すおよびやり直し機能を実装する方法 (詳細なチュートリアル)

Jun 02, 2018 am 11:54 AM
javascript

この記事では主に Immutable.js をベースにした undo と redo 機能と注意が必要な点を紹介します

ブラウザはますます強力になり、元々他のクライアントが提供していた多くの機能は徐々に移行しています。フロントエンドに至るまで、フロントエンド アプリケーションはますます複雑になっています。多くのフロントエンド アプリケーション、特に一部のオンライン編集ソフトウェアは、操作中にユーザー インタラクションを継続的に処理し、スムーズなインタラクションを保証するために元に戻す機能とやり直し機能を提供する必要があります。ただし、アプリケーションに元に戻す機能とやり直し機能を実装するのは簡単な作業ではありません。 Redux の公式ドキュメントでは、Redux アプリケーションに元に戻す機能とやり直し機能を実装する方法が紹介されています。 redux に基づく元に戻す機能はトップダウンのソリューションです。redux-undo の導入後は、すべての操作が「元に戻すことが可能」になり、その後も元に戻す機能をより使いやすくするために設定の変更を続けます (これも同様です) redux-undo 設定項目が多い理由)。

この記事では、シンプルなオンライン描画ツールを例として、TypeScript と Immutable.js を使用して実用的な「元に戻す」と「やり直し」機能を実装する、ボトムアップ アプローチを採用します。おおよその効果は以下の図に示されているとおりです:

ステップ 1: どの州に履歴レコードが必要かを判断し、カスタム State クラスを作成します

すべての州が履歴レコードを必要とするわけではありません。多くの状態、特にマウスやキーボードの操作に関連する状態は非常に簡単です。たとえば、描画ツールでグラフィックをドラッグする場合、「ドラッグ中」マークを設定する必要があり、ページには対応するドラッグ プロンプトが表示されます。当然、ドラッグ マークは履歴に表示されるべきではありません。また、Web ページのウィンドウ サイズ、バックグラウンドに送信されたリクエストのリストなど、他の状態は取り消すことができない、または取り消す必要がありません。

履歴レコードを必要としない状態を除外し、残りの状態を Immutable Record でカプセル化し、State クラスを定義します:

// State.ts
import { Record, List, Set } from 'immutable'
const StateRecord = Record({
 items: List<Item>
 transform: d3.ZoomTransform
 selection: number
})
// 用类封装,便于书写 TypeScript,注意这里最好使用Immutable 4.0 以上的版本
export default class State extends StateRecord {}
ログイン後にコピー

ここでの例は単純なオンライン描画ツールであるため、上記の State クラスには以下が含まれます3 つのフィールドは描画されたグラフィックを記録するために使用され、transform はアートボードのパンとズームのステータスを記録するために使用され、selection は現在選択されているグラフィックの ID を表します。グラフィック描画プレビュー、自動配置設定、操作プロンプト テキストなど、描画ツールの他の状態は State クラスには配置されません。

ステップ 2: Action 基本クラスを定義し、異なる操作ごとに対応する Action サブクラスを作成します

redux-undo との違いは、引き続きコマンド モードを使用することです。つまり、基本クラス Action と State All のすべてのペアを定義します。操作は Action のインスタンスとしてカプセル化され、さまざまなタイプの操作に対応する Action のいくつかのサブクラスが定義されます。

TypeScript では、Action 基本クラスを Abstract Class で定義する方が便利です。

// actions/index.ts
export default abstract class Action {
 abstract next(state: State): State
 abstract prev(state: State): State
 prepare(appHistory: AppHistory): AppHistory { return appHistory }
 getMessage() { return this.constructor.name }
}
ログイン後にコピー

Action オブジェクトの next メソッドは「次の状態」を計算するために使用され、prev メソッドは「前の状態」を計算するために使用されます。 getMessage メソッドは、Action オブジェクトの簡単な説明を取得するために使用されます。 getMessageメソッドを利用することで、ユーザーの操作記録をページ上に表示することができ、ユーザーは最近何が起こったのかを理解しやすくなります。 prepare メソッドは、アクションを初めて適用する前にアクションを「準備」するために使用されます。AppHistory の定義については、この記事で後ほど説明します。

Action サブクラスの例

次の AddItemAction は、「新しいグラフィックの追加」を表現するために使用される典型的な Action サブクラスです。

// actions/AddItemAction.ts
export default class AddItemAction extends Action {
 newItem: Item
 prevSelection: number
 constructor(newItem: Item) {
 super()
 this.newItem = newItem
 }
 prepare(history: AppHistory) {
 // 创建新的图形后会自动选中该图形,为了使得撤销该操作时 state.selection 变为原来的值
 // prepare 方法中读取了「添加图形之前 selection 的值」并保存到 this.prevSelection
 this.prevSelection = history.state.selection
 return history
 }
 next(state: State) {
 return state
  .setIn([&#39;items&#39;, this.newItem.id], this.newItem)
  .set(&#39;selection&#39;, this.newItemId)
 }
 prev(state: State) {
 return state
  .deleteIn([&#39;items&#39;, this.newItem.id])
  .set(&#39;selection&#39;, this.prevSelection)
 }
 getMessage() { return `Add item ${this.newItem.id}` }
}
ログイン後にコピー

実行時の動作

アプリケーションの実行中、ユーザーの操作により Action オブジェクトが生成されるたびに、オブジェクトの next メソッドを呼び出して次の状態を計算します。次に、アクションを後で使用できるようにリストに保存します。ユーザーが元に戻す操作を実行すると、アクション リストから最新のアクションを取得し、その prev メソッドを呼び出します。アプリケーションの実行中、next/prev メソッドは大まかに次のように呼び出されます:

// initState 是一开始就给定的应用初始状态
// 某一时刻,用户交互产生了 action1 ...
state1 = action1.next(initState)
// 又一个时刻,用户交互产生了 action2 ...
state2 = action2.next(state1)
// 同样的,action3也出现了 ...
state3 = action3.next(state2)
// 用户进行撤销,此时我们需要调用最近一个action的prev方法
state4 = action3.prev(state3)
// 如果再次进行撤销,我们从action列表中取出对应的action,调用其prev方法
state5 = action2.prev(state4)
// 重做的时候,取出最近一个被撤销的action,调用其next方法
state6 = action2.next(state5)
Applied-Action
ログイン後にコピー

以下の説明を容易にするために、Applied-Action を簡単に定義します。Applied-Action は、次の操作結果を指します。現在のアプリケーション状態のアクションに反映され、アクションの次のメソッドが実行されると、アクションは適用されます。前のメソッドが実行されると、アクションは適用されなくなります。

ステップ 3: 履歴コンテナ AppHistory を作成する

前の State クラスは、特定の時点でのアプリケーションの状態を表すために使用されます。 次に、アプリケーションの履歴を表す AppHistory クラスを定義します。同様に、履歴レコードを定義するために今でも Immutable Record を使用しています。状態フィールドは現在のアプリケーションのステータスを表すために使用され、リスト フィールドはすべてのアクションを保存するために使用され、インデックス フィールドは最新の適用されたアクションの添え字を記録するために使用されます。アプリケーションの履歴ステータスは、元に戻す/やり直しメソッドを通じて計算できます。 apply メソッドは、特定のアクションを AppHistory に追加して実行するために使用されます。具体的なコードは次のとおりです:

// AppHistory.ts
const emptyAction = Symbol(&#39;empty-action&#39;)
export const undo = Symbol(&#39;undo&#39;)
export type undo = typeof undo // TypeScript2.7之后对symbol的支持大大增强
export const redo = Symbol(&#39;redo&#39;)
export type redo = typeof redo
const AppHistoryRecord = Record({
 // 当前应用状态
 state: new State(),
 // action 列表
 list: List<Action>(),
 // index 表示最后一个applied-action在list中的下标。-1 表示没有任何applied-action
 index: -1,
})
export default class AppHistory extends AppHistoryRecord {
 pop() { // 移除最后一项操作记录
 return this
  .update(&#39;list&#39;, list => list.splice(this.index, 1))
  .update(&#39;index&#39;, x => x - 1)
 }
 getLastAction() { return this.index === -1 ? emptyAction : this.list.get(this.index) }
 getNextAction() { return this.list.get(this.index + 1, emptyAction) }
 apply(action: Action) {
 if (action === emptyAction) return this
 return this.merge({
  list: this.list.setSize(this.index + 1).push(action),
  index: this.index + 1,
  state: action.next(this.state),
 })
 }
 redo() {
 const action = this.getNextAction()
 if (action === emptyAction) return this
 return this.merge({
  list: this.list,
  index: this.index + 1,
  state: action.next(this.state),
 })
 }
 undo() {
 const action = this.getLastAction()
 if (action === emptyAction) return this
 return this.merge({
  list: this.list,
  index: this.index - 1,
  state: action.prev(this.state),
 })
 }
}
ログイン後にコピー

第四步:添加「撤销重做」功能

假设应用中的其他代码已经将网页上的交互转换为了一系列的 Action 对象,那么给应用添上「撤销重做」功能的大致代码如下:

type HybridAction = undo | redo | Action
// 如果用Redux来管理状态,那么使用下面的reudcer来管理那些「需要历史记录的状态」
// 然后将该reducer放在应用状态树中合适的位置
function reducer(history: AppHistory, action: HybridAction): AppHistory {
 if (action === undo) {
 return history.undo()
 } else if (action === redo) {
 return history.redo()
 } else { // 常规的 Action
 // 注意这里需要调用prepare方法,好让该action「准备好」
 return action.prepare(history).apply(action)
 }
}
// 如果是在 Stream/Observable 的环境下,那么像下面这样使用 reducer
const action$: Stream<HybridAction> = generatedFromUserInteraction
const appHistory$: Stream<AppHistory> = action$.fold(reducer, new AppHistory())
const state$ = appHistory$.map(h => h.state)
// 如果是用回调函数的话,大概像这样使用reducer
onActionHappen = function (action: HybridAction) {
 const nextHistory = reducer(getLastHistory(), action)
 updateAppHistory(nextHistory)
 updateState(nextHistory.state)
}
ログイン後にコピー

第五步:合并 Action,完善用户交互体验

通过上面这四个步骤,画图工具拥有了撤消重做功能,但是该功能用户体验并不好。在画图工具中拖动一个图形时,MoveItemAction 的产生频率和 mousemove 事件的发生频率相同,如果我们不对该情况进行处理,MoveItemAction 马上会污染整个历史记录。我们需要合并那些频率过高的 action,使得每个被记录下来的 action 有合理的撤销粒度。

每个 Action 在被应用之前,其 prepare 方法都会被调用,我们可以在 prepare 方法中对历史记录进行修改。例如,对于 MoveItemAction,我们判断上一个 action 是否和当前 action 属于同一次移动操作,然后来决定在应用当前 action 之前是否移除上一个 action。代码如下:

// actions/MoveItemAction.ts
export default class MoveItemAction extends Action {
 prevItem: Item
 // 一次图形拖动操作可以由以下三个变量来进行描述:
 // 拖动开始时鼠标的位置(startPos),拖动过程中鼠标的位置(movingPos),以及拖动的图形的 ID
 constructor(readonly startPos: Point, readonly movingPos: Point, readonly itemId: number) {
 // 上一行中 readonly startPos: Point 相当于下面两步:
 // 1. 在MoveItemAction中定义startPos只读字段
 // 2. 在构造函数中执行 this.startPos = startPos
 super()
 }
 prepare(history: AppHistory) {
 const lastAction = history.getLastAction()
 if (lastAction instanceof MoveItemAction && lastAction.startPos == this.startPos) {
  // 如果上一个action也是MoveItemAction,且拖动操作的鼠标起点和当前action相同
  // 则我们认为这两个action在同一次移动操作中
  this.prevItem = lastAction.prevItem
  return history.pop() // 调用pop方法来移除最近一个action
 } else {
  // 记录图形被移动之前的状态,用于撤销
  this.prevItem = history.state.items.get(this.itemId)
  return history
 }
 }
 next(state: State): State {
 const dx = this.movingPos.x - this.startPos.x
 const dy = this.movingPos.y - this.startPos.y
 const moved = this.prevItem.move(dx, dy)
 return state.setIn([&#39;items&#39;, this.itemId], moved)
 }
 prev(state: State) {
 // 撤销的时候我们直接使用已经保存的prevItem即可
 return state.setIn([&#39;items&#39;, this.itemId], this.prevItem)
 }
 getMessage() { /* ... */ }
}
ログイン後にコピー

从上面的代码中可以看到,prepare 方法除了使 action 自身准备好之外,它还可以让历史记录准备好。不同的 Action 类型有不同的合并规则,为每种 Action 实现合理的 prepare 函数之后,撤消重做功能的用户体验能够大大提升。

一些其他需要注意的地方

撤销重做功能是非常依赖于不可变性的,一个 Action 对象在放入 AppHistory.list 之后,其所引用的对象都应该是不可变的。如果 action 所引用的对象发生了变化,那么在后续撤销时可能发生错误。本方案中,为了方便记录操作发生时的一些必要信息,Action 对象的 prepare 方法中允许出现原地修改操作,但是 prepare 方法只会在 action 被放入历史记录之前调用一次,action 一旦进入纪录列表就是不可变的了。

上面是我整理给大家的,希望今后会对大家有帮助。

相关文章:

vue页面离开后执行函数的实例

vue轮播图插件vue-concise-slider的使用

vue加载自定义的js文件方法

以上がImmutable.js で元に戻すおよびやり直し機能を実装する方法 (詳細なチュートリアル)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

WebSocket と JavaScript を使用してオンライン音声認識システムを実装する方法 WebSocket と JavaScript を使用してオンライン音声認識システムを実装する方法 Dec 17, 2023 pm 02:54 PM

WebSocket と JavaScript を使用してオンライン音声認識システムを実装する方法 はじめに: 技術の継続的な発展により、音声認識技術は人工知能の分野の重要な部分になりました。 WebSocket と JavaScript をベースとしたオンライン音声認識システムは、低遅延、リアルタイム、クロスプラットフォームという特徴があり、広く使用されるソリューションとなっています。この記事では、WebSocket と JavaScript を使用してオンライン音声認識システムを実装する方法を紹介します。

WebSocket と JavaScript: リアルタイム監視システムを実装するための主要テクノロジー WebSocket と JavaScript: リアルタイム監視システムを実装するための主要テクノロジー Dec 17, 2023 pm 05:30 PM

WebSocketとJavaScript:リアルタイム監視システムを実現するためのキーテクノロジー はじめに: インターネット技術の急速な発展に伴い、リアルタイム監視システムは様々な分野で広く利用されています。リアルタイム監視を実現するための重要なテクノロジーの 1 つは、WebSocket と JavaScript の組み合わせです。この記事では、リアルタイム監視システムにおける WebSocket と JavaScript のアプリケーションを紹介し、コード例を示し、その実装原理を詳しく説明します。 1.WebSocketテクノロジー

WebSocketとJavaScriptを使ったオンライン予約システムの実装方法 WebSocketとJavaScriptを使ったオンライン予約システムの実装方法 Dec 17, 2023 am 09:39 AM

WebSocket と JavaScript を使用してオンライン予約システムを実装する方法 今日のデジタル時代では、ますます多くの企業やサービスがオンライン予約機能を提供する必要があります。効率的かつリアルタイムのオンライン予約システムを実装することが重要です。この記事では、WebSocket と JavaScript を使用してオンライン予約システムを実装する方法と、具体的なコード例を紹介します。 1. WebSocket とは何ですか? WebSocket は、単一の TCP 接続における全二重方式です。

JavaScript と WebSocket を使用してリアルタイムのオンライン注文システムを実装する方法 JavaScript と WebSocket を使用してリアルタイムのオンライン注文システムを実装する方法 Dec 17, 2023 pm 12:09 PM

JavaScript と WebSocket を使用してリアルタイム オンライン注文システムを実装する方法の紹介: インターネットの普及とテクノロジーの進歩に伴い、ますます多くのレストランがオンライン注文サービスを提供し始めています。リアルタイムのオンライン注文システムを実装するには、JavaScript と WebSocket テクノロジを使用できます。 WebSocket は、TCP プロトコルをベースとした全二重通信プロトコルで、クライアントとサーバー間のリアルタイム双方向通信を実現します。リアルタイムオンラインオーダーシステムにおいて、ユーザーが料理を選択して注文するとき

簡単な JavaScript チュートリアル: HTTP ステータス コードを取得する方法 簡単な JavaScript チュートリアル: HTTP ステータス コードを取得する方法 Jan 05, 2024 pm 06:08 PM

JavaScript チュートリアル: HTTP ステータス コードを取得する方法、特定のコード例が必要です 序文: Web 開発では、サーバーとのデータ対話が頻繁に発生します。サーバーと通信するとき、多くの場合、返された HTTP ステータス コードを取得して操作が成功したかどうかを判断し、さまざまなステータス コードに基づいて対応する処理を実行する必要があります。この記事では、JavaScript を使用して HTTP ステータス コードを取得する方法を説明し、いくつかの実用的なコード例を示します。 XMLHttpRequestの使用

JavaScript と WebSocket: 効率的なリアルタイム天気予報システムの構築 JavaScript と WebSocket: 効率的なリアルタイム天気予報システムの構築 Dec 17, 2023 pm 05:13 PM

JavaScript と WebSocket: 効率的なリアルタイム天気予報システムの構築 はじめに: 今日、天気予報の精度は日常生活と意思決定にとって非常に重要です。テクノロジーの発展に伴い、リアルタイムで気象データを取得することで、より正確で信頼性の高い天気予報を提供できるようになりました。この記事では、JavaScript と WebSocket テクノロジを使用して効率的なリアルタイム天気予報システムを構築する方法を学びます。この記事では、具体的なコード例を通じて実装プロセスを説明します。私たちは

JavaScript で HTTP ステータス コードを簡単に取得する方法 JavaScript で HTTP ステータス コードを簡単に取得する方法 Jan 05, 2024 pm 01:37 PM

JavaScript で HTTP ステータス コードを取得する方法の紹介: フロントエンド開発では、バックエンド インターフェイスとの対話を処理する必要があることが多く、HTTP ステータス コードはその非常に重要な部分です。 HTTP ステータス コードを理解して取得すると、インターフェイスから返されたデータをより適切に処理できるようになります。この記事では、JavaScript を使用して HTTP ステータス コードを取得する方法と、具体的なコード例を紹介します。 1. HTTP ステータス コードとは何ですか? HTTP ステータス コードとは、ブラウザがサーバーへのリクエストを開始したときに、サービスが

JavaScriptでinsertBeforeを使用する方法 JavaScriptでinsertBeforeを使用する方法 Nov 24, 2023 am 11:56 AM

使用法: JavaScript では、insertBefore() メソッドを使用して、DOM ツリーに新しいノードを挿入します。このメソッドには、挿入される新しいノードと参照ノード (つまり、新しいノードが挿入されるノード) の 2 つのパラメータが必要です。

See all articles