Я наткнулся на этот пост, пытаясь решить похожую проблему, и решил, что опубликую свой подход и решение для следующего парня.
Я придерживался вашего мнения - клонируем объект и пополняем старые данные на "отменить":
1) Скопируйте объект данных в новую переменную страницы ("_initData")
2) Создать Observable из исходного объекта сервера
3) при "отмене" перезагрузки, наблюдаемой с неизмененными данными ("_initData")
Упрощенный JS:
var _viewModel;
var _initData = {};
$(function () {
//on initial load
$.post("/loadMeUp", {}, function (data) {
$.extend(_initData , data);
_viewModel = ko.mapping.fromJS(data);
});
//to rollback changes
$("#undo").live("click", function (){
var data = {};
$.extend(data, _initData );
ko.mapping.fromJS(data, {}, _viewModel);
});
//when updating whole object from server
$("#updateFromServer).live("click", function(){
$.post("/loadMeUp", {}, function (data) {
$.extend(_initData , data);
ko.mapping.fromJS(data, {}, _viewModel);
});
});
//to just load a single item within the observable (for instance, nested objects)
$("#updateSpecificItemFromServer).live("click", function(){
$.post("/loadMeUpSpecificItem", {}, function (data) {
$.extend(_initData.SpecificItem, data);
ko.mapping.fromJS(data, {}, _viewModel.SpecificItem);
});
});
//updating subItems from both lists
$(".removeSpecificItem").live("click", function(){
//object id = "element_" + id
var id = this.id.split("_")[1];
$.post("/deleteSpecificItem", { itemID: id }, function(data){
//Table of items with the row elements id = "tr_" + id
$("#tr_" + id).remove();
$.each(_viewModel.SpecificItem.Members, function(index, value){
if(value.ID == id)
_viewModel.SpecificItem.Members.splice(index, 1);
});
$.each(_initData.SpecificItem.Members, function(index, value){
if(value.ID == id)
_initData.SpecificItem.Members.splice(index, 1);
});
});
});
});
У меня был достаточно сложный объект, поэтому я не хотел добавлять обработчики для каждого отдельного свойства.
Некоторые изменения сделаны в моем объекте в режиме реального времени, эти изменения редактируют как наблюдаемые, так и "_initData".
Когда я получаю данные с сервера, я обновляю свой объект "_initData", чтобы попытаться синхронизировать его с сервером.