home *** CD-ROM | disk | FTP | other *** search
- /* -*- Mode: Java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
- *
- * The contents of this file are subject to the Netscape Public
- * License Version 1.1 (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.mozilla.org/NPL/
- *
- * Software distributed under the License is distributed on an "AS
- * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
- * implied. See the License for the specific language governing
- * rights and limitations under the License.
- *
- * The Original Code is Mozilla Communicator client code.
- *
- * The Initial Developer of the Original Code is Netscape Communications
- * Corporation. Portions created by Netscape are
- * Copyright (C) 1998 Netscape Communications Corporation. All
- * Rights Reserved.
- *
- * Original Author:
- * Alec Flett <alecf@netscape.com>
- *
- * Contributor(s):
- * Seth Spitzer <sspitzer@netscape.com>
- */
-
- var gTotalSearchTerms=0;
- var gSearchRowContainer;
- var gSearchTerms = new Array;
- var gSearchRemovedTerms = new Array;
- var gSearchScope;
- var gSearchLessButton;
- var gSearchBooleanRadiogroup;
- var gSearchTermTree;
-
- //
- function searchTermContainer() {}
-
- searchTermContainer.prototype = {
- internalSearchTerm : '',
- internalBooleanAnd : '',
-
- // this.searchTerm: the actual nsIMsgSearchTerm object
- get searchTerm() { return this.internalSearchTerm; },
- set searchTerm(val) {
- this.internalSearchTerm = val;
-
- var term = val;
- // val is a nsIMsgSearchTerm
- var searchAttribute=this.searchattribute;
- var searchOperator=this.searchoperator;
- var searchValue=this.searchvalue;
-
- // now reflect all attributes of the searchterm into the widgets
- if (searchAttribute) searchAttribute.value = term.attrib;
- if (searchOperator) searchOperator.value = val.op;
- if (searchValue) searchValue.value = term.value;
-
- this.booleanAnd = val.booleanAnd;
- return val;
- },
-
- // searchscope - just forward to the searchattribute
- get searchScope() {
- if (this.searchattribute)
- return this.searchattribute.searchScope;
- return undefined;
- },
- set searchScope(val) {
- var searchAttribute = this.searchattribute;
- if (searchAttribute) searchAttribute.searchScope=val;
- return val;
- },
-
- saveId: function (element, slot) {
- this[slot] = element.id;
- },
-
- getElement: function (slot) {
- return document.getElementById(this[slot]);
- },
-
- // three well-defined properties:
- // searchattribute, searchoperator, searchvalue
- // the trick going on here is that we're storing the Element's Id,
- // not the element itself, because the XBL object may change out
- // from underneath us
- get searchattribute() { return this.getElement("internalSearchAttributeId"); },
- set searchattribute(val) {
- this.saveId(val, "internalSearchAttributeId");
- return val;
- },
- get searchoperator() { return this.getElement("internalSearchOperatorId"); },
- set searchoperator(val) {
- this.saveId(val, "internalSearchOperatorId");
- return val;
- },
- get searchvalue() { return this.getElement("internalSearchValueId"); },
- set searchvalue(val) {
- this.saveId(val, "internalSearchValueId");
- return val;
- },
-
- booleanNodes: null,
- stringBundle: document.getElementById("bundle_search"),
- get booleanAnd() { return this.internalBooleanAnd; },
- set booleanAnd(val) {
- // whenever you set this, all nodes in booleanNodes
- // are updated to reflect the string
-
- if (this.internalBooleanAnd == val) return val;
- this.internalBooleanAnd = val;
-
- var booleanNodes = this.booleanNodes;
- if (!booleanNodes) return val;
-
- var stringBundle = this.stringBundle;
- var andString = val ? "And" : "Or";
- for (var i=0; i<booleanNodes.length; i++) {
- try {
- var staticString =
- stringBundle.getString("search" + andString + i);
- if (staticString && staticString.length>0)
- booleanNodes[i].setAttribute("label", staticString);
- } catch (ex) { /* no error, means string not found */}
- }
- return val;
- },
-
- save: function () {
- var searchTerm = this.searchTerm;
- searchTerm.attrib = this.searchattribute.value;
- searchTerm.op = this.searchoperator.value;
- if (this.searchvalue.value)
- this.searchvalue.save();
- else
- this.searchvalue.saveTo(searchTerm.value);
- searchTerm.value = this.searchvalue.value;
- searchTerm.booleanAnd = this.booleanAnd;
- },
- // if you have a search term element with no search term
- saveTo: function(searchTerm) {
- this.internalSearchTerm = searchTerm;
- this.save();
- }
- }
-
- var nsIMsgSearchTerm = Components.interfaces.nsIMsgSearchTerm;
-
- function initializeSearchWidgets() {
- gSearchBooleanRadiogroup = document.getElementById("booleanAndGroup");
- gSearchRowContainer = document.getElementById("searchTermList");
- gSearchTermTree = document.getElementById("searchTermTree");
- gSearchLessButton = document.getElementById("less");
- if (!gSearchLessButton)
- dump("I couldn't find less button!");
- }
-
- function initializeBooleanWidgets()
- {
- var booleanAnd = true;
- // get the boolean value from the first term
- var firstTerm = gSearchTerms[0].searchTerm;
- if (firstTerm)
- booleanAnd = firstTerm.booleanAnd;
-
- // target radio items have value="and" or value="or"
- var targetValue = "or";
- if (booleanAnd) targetValue = "and";
-
- var targetElement = gSearchBooleanRadiogroup.getElementsByAttribute("value", targetValue)[0];
-
- gSearchBooleanRadiogroup.selectedItem = targetElement;
- }
-
- function initializeSearchRows(scope, searchTerms)
- {
- gTotalSearchTerms = searchTerms.Count();
- for (var i=0; i<gTotalSearchTerms; i++) {
- var searchTerm = searchTerms.QueryElementAt(i, nsIMsgSearchTerm);
- createSearchRow(i, scope, searchTerm);
- }
- initializeBooleanWidgets();
- }
-
- function scrollToLastSearchTerm(index)
- {
- if (index > 0)
- gSearchTermTree.ensureIndexIsVisible(index-1);
- }
-
- function onMore(event)
- {
- if(gTotalSearchTerms==1)
- gSearchLessButton .removeAttribute("disabled", "false");
- createSearchRow(gTotalSearchTerms++, gSearchScope, null);
- // the user just added a term, so scroll to it
- scrollToLastSearchTerm(gTotalSearchTerms);
- }
-
- function onLess(event)
- {
- if (gTotalSearchTerms>1)
- removeSearchRow(--gTotalSearchTerms);
- if (gTotalSearchTerms==1)
- gSearchLessButton .setAttribute("disabled", "true");
-
- // the user removed a term, so scroll to the bottom so they are aware of it
- scrollToLastSearchTerm(gTotalSearchTerms);
- }
-
- // set scope on all visible searchattribute tags
- function setSearchScope(scope) {
- gSearchScope = scope;
- for (var i=0; i<gSearchTerms.length; i++) {
- gSearchTerms[i].obj.searchattribute.searchScope = scope;
- gSearchTerms[i].scope = scope;
- }
- }
-
- function booleanChanged(event) {
- // when boolean changes, we have to update all the attributes on the
- // search terms
-
- var newBoolValue =
- (event.target.getAttribute("value") == "and") ? true : false;
- for (var i=0; i<gSearchTerms.length; i++) {
- var searchTerm = gSearchTerms[i].obj;
- searchTerm.booleanAnd = newBoolValue;
- }
- }
-
-
- function createSearchRow(index, scope, searchTerm)
- {
- var searchAttr = document.createElement("searchattribute");
- var searchOp = document.createElement("searchoperator");
- var searchVal = document.createElement("searchvalue");
-
- // now set up ids:
- searchAttr.id = "searchAttr" + index;
- searchOp.id = "searchOp" + index;
- searchVal.id = "searchVal" + index;
-
- searchAttr.setAttribute("for", searchOp.id + "," + searchVal.id);
-
- var rowdata = new Array(null, searchAttr,
- null, searchOp,
- null, searchVal,
- null);
- var searchrow = constructRow(rowdata);
- searchrow.id = "searchRow" + index;
-
- var searchTermObj = new searchTermContainer;
-
- gSearchTerms[gSearchTerms.length] = {obj:searchTermObj, scope:scope, searchTerm:searchTerm, initialized:false};
-
- searchTermObj.searchattribute = searchAttr;
- searchTermObj.searchoperator = searchOp;
- searchTermObj.searchvalue = searchVal;
-
- // now invalidate the newly created items because they've been inserted
- // into the document, and XBL bindings will be inserted in their place
- //searchAttr = searchOp = searchVal = undefined;
-
- // and/or string handling:
- // this is scary - basically we want to take every other
- // treecell, (note the i+=2) which will be a text label,
- // and set the searchTermObj's
- // booleanNodes to that
- var stringNodes = new Array;
- var treecells = searchrow.firstChild.childNodes;
- var j=0;
- for (var i=0; i<treecells.length; i+=2) {
- stringNodes[j++] = treecells[i];
- }
- searchTermObj.booleanNodes = stringNodes;
-
- gSearchRowContainer.appendChild(searchrow);
- }
-
- function initializeTermFromId(id)
- {
- // id is of the form searchAttr<n>
- // strlen("searchAttr") == 10
- // turn searchAttr<n> -> <n>
- var index = parseInt(id.slice(10));
- initializeTermFromIndex(index)
- }
-
- function initializeTermFromIndex(index)
- {
- var searchTermObj = gSearchTerms[index].obj;
-
- searchTermObj.searchScope = gSearchTerms[index].scope;
- // the search term will initialize the searchTerm element, including
- // .booleanAnd
- if (gSearchTerms[index].searchTerm)
- searchTermObj.searchTerm = gSearchTerms[index].searchTerm;
- // here, we don't have a searchTerm, so it's probably a new element -
- // we'll initialize the .booleanAnd from the existing setting in
- // the UI
- else
- searchTermObj.booleanAnd = getBooleanAnd();
-
- gSearchTerms[index].initialized = true;
- }
-
- // creates a <treerow> using the array treeCellChildren as
- // the children of each treecell
- function constructRow(treeCellChildren)
- {
- var treeitem = document.createElement("treeitem");
- var row = document.createElement("treerow");
- for (var i = 0; i<treeCellChildren.length; i++) {
- var treecell = document.createElement("treecell");
-
- // it's ok to have empty cells
- if (treeCellChildren[i]) {
- treecell.setAttribute("allowevents", "true");
- treeCellChildren[i].setAttribute("flex", "1");
- treecell.appendChild(treeCellChildren[i]);
- }
- row.appendChild(treecell);
- }
- treeitem.appendChild(row);
- return treeitem;
- }
-
- function removeSearchRow(index)
- {
- var searchTermObj = gSearchTerms[index].obj;
- if (!searchTermObj) {
- return;
- }
-
- // if it is an existing (but offscreen) term,
- // make sure it is initialized before we remove it.
- if (!gSearchTerms[index].searchTerm && !gSearchTerms[index].initialized)
- initializeTermFromIndex(index);
-
- // need to remove row from tree, so walk upwards from the
- // searchattribute to find the first <treeitem>
- var treeItemRow = searchTermObj.searchattribute;
- while (treeItemRow) {
- if (treeItemRow.localName == "treeitem") break;
- treeItemRow = treeItemRow.parentNode;
- }
-
- if (!treeItemRow) {
- dump("Error: couldn't find parent treeitem!\n");
- return;
- }
-
-
- if (searchTermObj.searchTerm) {
- gSearchRemovedTerms[gSearchRemovedTerms.length] = searchTermObj.searchTerm;
- } else {
- //dump("That wasn't real. ignoring \n");
- }
-
- treeItemRow.parentNode.removeChild(treeItemRow);
- // remove it from the list of terms - XXX this does it?
- // remove the last element
- gSearchTerms.length--;
- }
-
- function getBooleanAnd()
- {
- if (gSearchBooleanRadiogroup.selectedItem)
- return (gSearchBooleanRadiogroup.selectedItem.getAttribute("value") == "and") ? true : false;
-
- // default to false
- return false;
- }
-
- // save the search terms from the UI back to the actual search terms
- // searchTerms: nsISupportsArray of terms
- // termOwner: object which can contain and create the terms
- // (will be unnecessary if we just make terms creatable
- // via XPCOM)
- function saveSearchTerms(searchTerms, termOwner)
- {
- var i;
- for (i = 0; i<gSearchTerms.length; i++) {
- try {
- var searchTerm = gSearchTerms[i].obj.searchTerm;
-
- // the term might be an offscreen one we haven't initialized yet
- // if so, don't bother saving it.
- if (!searchTerm && !gSearchTerms[i].initialized) {
- // is an existing term, but not initialize, so skip saving
- continue;
- }
-
- if (searchTerm)
- gSearchTerms[i].obj.save();
- else {
- // need to create a new searchTerm, and somehow save it to that
- searchTerm = termOwner.createTerm();
- gSearchTerms[i].obj.saveTo(searchTerm);
- termOwner.appendTerm(searchTerm);
- }
- } catch (ex) {
- dump("** Error saving element " + i + ": " + ex + "\n");
- }
- }
-
- // now remove the queued elements
- for (i=0; i<gSearchRemovedTerms.length; i++) {
- // this is so nasty, we have to iterate through
- // because GetIndexOf is acting funny
- var searchTermSupports =
- gSearchRemovedTerms[i].QueryInterface(Components.interfaces.nsISupports);
- searchTerms.RemoveElement(searchTermSupports);
- }
-
- }
-
- function onReset(event)
- {
- while (gTotalSearchTerms>0)
- removeSearchRow(--gTotalSearchTerms);
- onMore(event);
- }
-
- function convertPRTimeToString(tm)
- {
- var time = new Date();
- // PRTime is in microseconds, Javascript time is in seconds
- // so divide by 1000 when converting
- time.setTime(tm / 1000);
-
- return convertDateToString(time);
- }
-
- function convertDateToString(time)
- {
- var dateStr = time.getMonth() + 1;
- dateStr += "/";
- dateStr += time.getDate();
- dateStr += "/";
- dateStr += 1900 + time.getYear();
- return dateStr;
- }
-
- function convertStringToPRTime(str)
- {
- var time = new Date();
- time.setTime(Date.parse(str));
- // Javascript time is in seconds, PRTime is in microseconds
- // so multiply by 1000 when converting
- return (time.getTime() * 1000);
- }
-