// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy
// of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
// License for the specific language governing permissions and limitations under
// the License.
/*
* Page class for browse/index.html
*/
function CouchIndexPage() {
page = this;
this.addDatabase = function() {
$.showDialog("_create_database.html", {
submit: function(data) {
if (!data.name || data.name.length == 0) {
return {name: "Please enter a name."};
}
try {
new CouchDB(data.name).createDb();
} catch (e) {
return {name: e.reason};
}
if (window !== parent) parent.setTimeout("updateDatabaseList()", 500);
window.open("database.html?" + data.name, "content");
}
});
return false;
}
this.updateDatabaseListing = function() {
var allDbs = CouchDB.allDbs();
for (var i = 0; i < allDbs.length; i++) {
var dbName = allDbs[i];
var info = new CouchDB(dbName).info();
$("#databases tbody.content").append(
"
" +
dbName + " " + info.doc_count +" " +
info.update_seq + " ");
$("#databases tbody tr:odd").addClass("odd");
$("#databases tbody.footer tr td").text(allDbs.length + " database(s)");
}
}
}
/*
* Page class for browse/database.html
*/
function CouchDatabasePage() {
var urlParts = location.search.substr(1).split("/");
var dbName = urlParts.shift();
var viewName = (urlParts.length > 0) ? urlParts.join("/") : null;
if (!viewName) {
viewName = $.cookies.get(dbName + ".view") || "";
} else {
$.cookies.set(dbName + ".view", viewName);
}
var db = new CouchDB(dbName);
this.dbName = dbName;
this.viewName = viewName;
this.db = db;
this.isDirty = false;
page = this;
this.addDocument = function() {
$.showDialog("_create_document.html", {
submit: function(data) {
try {
var result = db.save(data.docid ? {_id: data.docid} : {});
} catch (err) {
return {docid: err.reason};
}
location.href = "document.html?" + dbName + "/" + result.id;
}
});
}
this.deleteDatabase = function() {
$.showDialog("_delete_database.html", {
submit: function() {
db.deleteDb();
location.href = "index.html";
if (window !== null) {
parent.$("#dbs li").filter(function(index) {
return $("a", this).text() == dbName;
}).remove();
}
}
});
}
this.populateViewEditor = function() {
if (viewName.match(/^_design\//)) {
page.revertViewChanges();
var dirtyTimeout = null;
function updateDirtyState() {
clearTimeout(dirtyTimeout);
dirtyTimeout = setTimeout(function() {
var buttons = $("#viewcode button.save, #viewcode button.revert");
page.isDirty = $("#viewcode textarea").val() != page.storedViewCode;
if (page.isDirty) {
buttons.removeAttr("disabled");
} else {
buttons.attr("disabled", "disabled");
}
}, 100);
}
$("#viewcode textarea").bind("input", updateDirtyState);
if ($.browser.msie) { // sorry, browser detection
$("#viewcode textarea").get(0).onpropertychange = updateDirtyState
} else if ($.browser.safari) {
$("#viewcode textarea").bind("paste", updateDirtyState)
.bind("change", updateDirtyState)
.bind("keydown", updateDirtyState)
.bind("keypress", updateDirtyState)
.bind("keyup", updateDirtyState)
.bind("textInput", updateDirtyState);
}
}
}
this.populateViewsMenu = function() {
var designDocs = db.allDocs({startkey: "_design/", endkey: "_design/ZZZ"});
$("#switch select").each(function() {
this.options.length = 3;
for (var i = 0; i < designDocs.rows.length; i++) {
var doc = db.open(designDocs.rows[i].id);
var optGroup = $(" ").attr("label", doc._id.substr(8));
for (var name in doc.views) {
if (!doc.views.hasOwnProperty(name)) continue;
$(" ").attr("value", doc._id + "/" + name).text(name)
.appendTo(optGroup);
}
optGroup.appendTo(this);
}
this.autocomplete = false;
});
}
this.revertViewChanges = function() {
if (!page.storedViewCode) {
var viewNameParts = viewName.split("/");
var designDocId = viewNameParts[1];
var localViewName = viewNameParts[2];
var designDoc = db.open(["_design", designDocId].join("/"));
if (designDoc) {
page.storedViewCode = designDoc.views[localViewName];
}
}
$("#viewcode textarea").val(page.storedViewCode);
page.isDirty = false;
$("#viewcode button.revert, #viewcode button.save").attr("disabled", "disabled");
}
this.saveViewAs = function() {
if (viewName && /^_design/.test(viewName)) {
var viewNameParts = viewName.split("/");
var designDocId = viewNameParts[1];
var localViewName = viewNameParts[2];
} else {
var designDocId = "", localViewName = ""
}
$.showDialog("_save_view_as.html", {
load: function(elem) {
$("#input_docid", elem).val(designDocId).suggest(function(text, callback) {
var matches = [];
var docs = db.allDocs({
count: 10, startkey: "_design/" + text,
endkey: "_design/" + text + "ZZZZ"
});
for (var i = 0; i < docs.rows.length; i++) {
matches[i] = docs.rows[i].id.substr(8);
}
callback(matches);
});
$("#input_name", elem).val(localViewName).suggest(function(text, callback) {
var matches = [];
try {
var doc = db.open("_design/" + $("#input_docid").val());
} catch (err) {
return;
}
if (!doc || !doc.views) return;
for (var viewName in doc.views) {
if (!doc.views.hasOwnProperty(viewName) || !viewName.match("^" + text)) {
continue;
}
matches.push(viewName);
}
callback(matches);
});
},
submit: function(data) {
if (!data.docid || !data.name) {
var errors = {};
if (!data.docid) errors.docid = "Please enter a document ID";
if (!data.name) errors.name = "Please enter a view name";
return errors;
}
var viewCode = $("#viewcode textarea").val();
var docId = ["_design", data.docid].join("/");
var designDoc = db.open(docId);
if (!designDoc) designDoc = {_id: docId, language: "text/javascript"};
if (designDoc.views === undefined) designDoc.views = {};
designDoc.views[data.name] = viewCode;
db.save(designDoc);
page.isDirty = false;
location.href = "database.html?" + dbName + "/" + designDoc._id + "/" + data.name;
}
});
}
this.saveViewChanges = function() {
var viewNameParts = viewName.split("/");
var designDocId = viewNameParts[1];
var localViewName = viewNameParts[2];
var designDoc = db.open(["_design", designDocId].join("/"));
var viewCode = $("#viewcode textarea").val();
designDoc.views[localViewName] = viewCode;
db.save(designDoc);
page.isDirty = false;
$("#viewcode button.revert, #viewcode button.save").attr("disabled", "disabled");
}
this.updateDesignDocLink = function() {
if (viewName && /^_design/.test(viewName)) {
var docId = "_design/" + viewName.split("/")[1];
$("#designdoc-link").attr("href", "document.html?" + dbName + "/" + docId).text(docId);
} else {
$("#designdoc-link").removeAttr("href").text("");
}
}
this.updateDocumentListing = function(options) {
if (options === undefined) options = {};
if (options.count === undefined) {
options.count = parseInt($("#perpage").val(), 10);
}
if ($("#documents thead th.key").is(".desc")) {
options.descending = true;
$.cookies.set(dbName + ".desc", "1");
} else {
if (options.descending !== undefined) delete options.descending;
$.cookies.remove(dbName + ".desc");
}
$("#paging a").unbind();
$("#documents tbody.content").empty();
this.updateDesignDocLink();
var result = null;
if (!viewName) {
$("#switch select").get(0).selectedIndex = 0;
result = db.allDocs(options);
} else {
$("#switch select").each(function() {
for (var i = 0; i < this.options.length; i++) {
if (this.options[i].value == viewName) {
this.selectedIndex = i;
break;
}
}
});
docs = [];
if (viewName == "_temp_view") {
$("#viewcode").show().addClass("expanded");
var query = $("#viewcode textarea").val();
$.cookies.set(db.name + ".query", query);
try {
result = db.query(query, options);
} catch (e) {
alert(e.reason ? e.reason : e.message);
return;
}
} else if (viewName == "_design_docs") {
options.startkey = options.descending ? "_design/ZZZZ" : "_design/";
options.endkey = options.descending ? "_design/" : "_design/ZZZZ";
result = db.allDocs(options);
} else {
$("#viewcode").show();
var currentViewCode = $("#viewcode textarea").val();
if (currentViewCode != page.storedViewCode) {
result = db.query(currentViewCode, options);
} else {
result = db.view(viewName.substr(8), options);
}
}
}
if (result.offset === undefined) {
result.offset = 0;
}
if (result.offset > 0) {
$("#paging a.prev").attr("href", "#" + (result.offset - options.count)).click(function() {
var firstDoc = result.rows[0];
page.updateDocumentListing({
startkey: firstDoc.key !== undefined ? firstDoc.key : null,
startkey_docid: firstDoc.id,
skip: 1,
count: -options.count
});
return false;
});
} else {
$("#paging a.prev").removeAttr("href");
}
if (result.total_rows - result.offset > options.count) {
$("#paging a.next").attr("href", "#" + (result.offset + options.count)).click(function() {
var lastDoc = result.rows[result.rows.length - 1];
page.updateDocumentListing({
startkey: lastDoc.key !== undefined ? lastDoc.key : null,
startkey_docid: lastDoc.id,
skip: 1,
count: options.count
});
return false;
});
} else {
$("#paging a.next").removeAttr("href");
}
for (var i = 0; i < result.rows.length; i++) {
var row = result.rows[i];
var tr = $(" ");
var key = row.key;
$("" +
"ID: " + row.id +
" ").find("em").text(
key !== null ? prettyPrintJSON(key, 0, "") : "null"
).end().appendTo(tr);
var value = row.value;
$(" ").text(
value !== null ? prettyPrintJSON(value, 0, "") : "null"
).appendTo(tr).dblclick(function() {
location.href = this.previousSibling.firstChild.href;
});
tr.appendTo("#documents tbody.content");
}
$("#documents tbody tr:odd").addClass("odd");
$("#documents tbody.footer td span").text(
"Showing " + Math.min(result.total_rows, result.offset + 1) + "-" +
(result.offset + result.rows.length) + " of " + result.total_rows +
" document" + (result.total_rows != 1 ? "s" : ""));
}
window.onbeforeunload = function() {
$("#switch select").val(viewName);
if (page.isDirty) {
return "You've made changes to the view code that have not been " +
"saved yet.";
}
}
}
/*
* Page class for browse/database.html
*/
function CouchDocumentPage() {
var urlParts = location.search.substr(1).split("/");
var dbName = urlParts.shift();
var idParts = urlParts.join("/").split("@", 2);
var docId = idParts[0];
var docRev = (idParts.length > 1) ? idParts[1] : null;
var db = new CouchDB(dbName);
var doc = db.open(docId, {revs_info: true});
var revs = doc._revs_info;
delete doc._revs_info;
if (docRev != null) {
try {
doc = db.open(docId, {rev: docRev});
} catch (e) {
alert("The requested revision was not found. " +
"You will be redirected back to the latest revision.");
location.href = "?" + dbName + "/" + docId;
return;
}
}
this.dbName = dbName;
this.db = db;
this.doc = doc;
this.isDirty = false;
page = this;
this.addField = function() {
var fieldName = "unnamed";
var fieldIdx = 1;
while (doc.hasOwnProperty(fieldName)) {
fieldName = "unnamed " + fieldIdx++;
}
doc[fieldName] = null;
var row = _addRowForField(fieldName);
page.isDirty = true;
_editKey(row.find("th"), fieldName);
}
this.updateFieldListing = function() {
$("#fields tbody.content").empty();
var propNames = [];
for (var prop in doc) {
if (!doc.hasOwnProperty(prop)) continue;
propNames.push(prop);
}
// Order properties alphabetically, but put internal fields first
propNames.sort(function(a, b) {
var a0 = a.charAt(0), b0 = b.charAt(0);
if (a0 == "_" && b0 != "_") {
return -1;
} else if (a0 != "_" && b0 == "_") {
return 1;
} else {
return a < b ? -1 : a != b ? 1 : 0;
}
});
for (var pi = 0; pi < propNames.length; pi++) {
_addRowForField(propNames[pi]);
}
if (revs.length > 1) {
var currentIndex = 0;
for (var i = 0; i < revs.length; i++) {
if (revs[i].rev == doc._rev) {
currentIndex = i;
break;
}
}
if (currentIndex < revs.length - 1) {
var prevRev = revs[currentIndex + 1].rev;
$("#paging a.prev").attr("href", "?" + dbName + "/" + docId + "@" + prevRev);
}
if (currentIndex > 0) {
var nextRev = revs[currentIndex - 1].rev;
$("#paging a.next").attr("href", "?" + dbName + "/" + docId + "@" + nextRev);
}
$("#fields tbody.footer td span").text("Showing revision " +
(revs.length - currentIndex) + " of " + revs.length);
}
}
this.deleteDocument = function() {
$.showDialog("_delete_document.html", {
submit: function() {
db.deleteDoc(doc);
location.href = "database.html?" + dbName;
}
});
}
this.saveDocument = function() {
try {
db.save(doc);
} catch (e) {
alert(e.reason);
return;
}
page.isDirty = false;
location.href = "?" + dbName + "/" + docId;
}
window.onbeforeunload = function() {
if (page.isDirty) {
return "You've made changes to this document that have not been " +
"saved yet.";
}
}
function _addRowForField(fieldName) {
var value = _renderValue(doc[fieldName]);
var row = $(" ")
.find("th").append($(" ").text(fieldName)).dblclick(function() {
_editKey(this, $(this).text());
}).end()
.find("td").append(value).dblclick(function() {
_editValue(this, $(this).prev("th").text());
}).end()
.appendTo("#fields tbody.content");
if (fieldName != "_id" && fieldName != "_rev") {
row.find("th, td").attr("title", "Double click to edit");
_initKey(row, fieldName);
_initValue(value);
}
$("#fields tbody tr").removeClass("odd").filter(":odd").addClass("odd");
return row;
}
function _editKey(cell, fieldName) {
if (fieldName == "_id" || fieldName == "_rev") return;
var th = $(cell);
th.empty();
var input = $(" ");
input.dblclick(function() { return false; }).keydown(function(evt) {
switch (evt.keyCode) {
case 13: applyChange(); break;
case 27: cancelChange(); break;
}
});
var tools = $("
");
function applyChange() {
input.nextAll().remove();
var newName = input.val();
if (!newName.length || newName == fieldName) {
cancelChange();
return;
}
doc[newName] = doc[fieldName];
delete doc[fieldName];
th.children().remove();
th.append($(" ").text(newName));
_initKey(th.parent("tr"), fieldName);
page.isDirty = true;
}
function cancelChange() {
th.children().remove();
th.append($(" ").text(fieldName));
_initKey(th.parent("tr"), fieldName);
}
$(" ").click(function() {
applyChange();
}).appendTo(tools);
$(" ").click(function() {
cancelChange();
}).appendTo(tools);
tools.appendTo(th);
input.val(fieldName).appendTo(th);
input.each(function() { this.focus(); this.select(); });
}
function _editValue(cell, fieldName) {
if (fieldName == "_id" || fieldName == "_rev") return;
var td = $(cell);
var value = doc[fieldName];
var needsTextarea = $("dl", td).length > 0 || $("code", td).text().length > 60;
td.empty();
if (needsTextarea) {
var input = $("");
} else {
var input = $(" ");
}
input.dblclick(function() { return false; }).keydown(function(evt) {
switch (evt.keyCode) {
case 13: if (!needsTextarea) applyChange(); break;
case 27: cancelChange(); break;
}
});
var tools = $("
");
function applyChange() {
input.nextAll().remove();
try {
var newValue = input.val() || "null";
if (newValue == doc[fieldName]) {
cancelChange();
return;
}
doc[fieldName] = JSON.parse(newValue);
td.children().remove();
page.isDirty = true;
var value = _renderValue(doc[fieldName]);
td.append(value);
_initValue(value);
} catch (err) {
input.addClass("invalid");
var msg = err.message;
if (msg == "parseJSON") {
msg = "Please enter a valid JSON value (for example, \"string\").";
}
$("
").text(msg).insertAfter(input);
}
}
function cancelChange() {
td.children().remove();
var value = _renderValue(doc[fieldName]);
td.append(value);
_initValue(value);
}
$(" ").click(function() {
applyChange();
}).appendTo(tools);
$(" ").click(function() {
cancelChange();
}).appendTo(tools);
tools.appendTo(td);
input.val(prettyPrintJSON(value)).appendTo(td);
input.each(function() { this.focus(); this.select(); });
if (needsTextarea) input.resizable();
}
function _initKey(row, fieldName) {
if (fieldName != "_id" && fieldName != "_rev") {
$(" ").click(function() {
delete doc[fieldName];
row.remove();
page.isDirty = true;
$("#fields tbody tr").removeClass("odd").filter(":odd").addClass("odd");
}).prependTo(row.find("th"));
}
}
function _initValue(value) {
value.find("dd").filter(":has(dl)").hide().prev("dt").addClass("collapsed");
value.find("dd").not(":has(dl)").addClass("inline").prev().addClass("inline");
value.find("dt.collapsed").click(function() {
$(this).toggleClass("collapsed").next().toggle();
});
}
function _renderValue(value) {
var type = typeof(value);
if (type == "object" && value !== null) {
var list = $(" ");
for (var i in value) {
if (!value.hasOwnProperty(i)) continue;
$(" ").text(i).appendTo(list);
$(" ").append(_renderValue(value[i])).appendTo(list);
}
return list;
} else {
return $("
").addClass(type).text(
value !== null ? JSON.stringify(value) : "null"
);
}
}
}