2038 lines
		
	
	
		
			69 KiB
		
	
	
	
		
			JavaScript
		
	
	
			
		
		
	
	
			2038 lines
		
	
	
		
			69 KiB
		
	
	
	
		
			JavaScript
		
	
	
(function(root, factory) {
 | 
						|
 | 
						|
  // Set up Backform appropriately for the environment. Start with AMD.
 | 
						|
  if (typeof define === 'function' && define.amd) {
 | 
						|
    define([
 | 
						|
      'underscore', 'underscore.string', 'jquery', 'backbone', 'backform',
 | 
						|
      'backgrid', 'codemirror', 'pgadmin.backgrid', 'codemirror/mode/sql/sql',
 | 
						|
      'select2'
 | 
						|
      ],
 | 
						|
     function(_, S, $, Backbone, Backform, Backgrid, CodeMirror) {
 | 
						|
      // Export global even in AMD case in case this script is loaded with
 | 
						|
      // others that may still expect a global Backform.
 | 
						|
      return factory(root, _, S, $, Backbone, Backform, Backgrid, CodeMirror);
 | 
						|
    });
 | 
						|
 | 
						|
  // Next for Node.js or CommonJS. jQuery may not be needed as a module.
 | 
						|
  } else if (typeof exports !== 'undefined') {
 | 
						|
    var _ = require('underscore') || root._,
 | 
						|
      $ = root.jQuery || root.$ || root.Zepto || root.ender,
 | 
						|
      Backbone = require('backbone') || root.Backbone,
 | 
						|
      Backform = require('backform') || root.Backform,
 | 
						|
      Backgrid = require('backgrid') || root.Backgrid;
 | 
						|
      CodeMirror = require('codemirror') || root.CodeMirror;
 | 
						|
      pgAdminBackgrid = require('pgadmin.backgrid');
 | 
						|
      S = require('underscore.string');
 | 
						|
    factory(root, _, S, $, Backbone, Backform, Backgrid, CodeMirror);
 | 
						|
 | 
						|
  // Finally, as a browser global.
 | 
						|
  } else {
 | 
						|
    factory(root, root._, root.s, (root.jQuery || root.Zepto || root.ender || root.$), root.Backbone, root.Backform, root.Backgrid, root.CodeMirror);
 | 
						|
  }
 | 
						|
}(this, function(root, _, S, $, Backbone, Backform, Backgrid, CodeMirror) {
 | 
						|
 | 
						|
  var pgAdmin = (window.pgAdmin = window.pgAdmin || {});
 | 
						|
 | 
						|
  pgAdmin.editableCell = function() {
 | 
						|
    if (this.attributes && !_.isUndefined(this.attributes.disabled) &&
 | 
						|
        !_.isNull(this.attributes.disabled)) {
 | 
						|
      if(_.isFunction(this.attributes.disabled)) {
 | 
						|
        return !(this.attributes.disabled.apply(this, arguments));
 | 
						|
      }
 | 
						|
      if (_.isBoolean(this.attributes.disabled)) {
 | 
						|
        return !this.attributes.disabled;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  // HTML markup global class names. More can be added by individual controls
 | 
						|
  // using _.extend. Look at RadioControl as an example.
 | 
						|
  _.extend(Backform, {
 | 
						|
    controlLabelClassName: "control-label col-sm-4",
 | 
						|
    controlsClassName: "pgadmin-controls col-sm-8",
 | 
						|
    groupClassName: "pgadmin-control-group form-group col-xs-12",
 | 
						|
    setGroupClassName: "set-group col-xs-12",
 | 
						|
    tabClassName: "backform-tab col-xs-12",
 | 
						|
    setGroupContentClassName: "fieldset-content col-xs-12"
 | 
						|
    });
 | 
						|
 | 
						|
  var controlMapper = Backform.controlMapper = {
 | 
						|
    'int': ['uneditable-input', 'integer', 'integer'],
 | 
						|
    'text': ['uneditable-input', 'input', 'string'],
 | 
						|
    'numeric': ['uneditable-input', 'numeric', 'numeric'],
 | 
						|
    'date': 'datepicker',
 | 
						|
    'boolean': 'boolean',
 | 
						|
    'options': ['readonly-option', 'select', Backgrid.Extension.PGSelectCell],
 | 
						|
    'multiline': ['textarea', 'textarea', 'string'],
 | 
						|
    'collection': ['sub-node-collection', 'sub-node-collection', 'string'],
 | 
						|
    'uniqueColCollection': ['unique-col-collection', 'unique-col-collection', 'string'],
 | 
						|
    'switch' : 'switch',
 | 
						|
    'select2': 'select2',
 | 
						|
    'note': 'note'
 | 
						|
  };
 | 
						|
 | 
						|
  var getMappedControl = Backform.getMappedControl = function(type, mode) {
 | 
						|
    if (type in Backform.controlMapper) {
 | 
						|
      var m = Backform.controlMapper[type];
 | 
						|
 | 
						|
      if (!_.isArray(m)) {
 | 
						|
        return m;
 | 
						|
      }
 | 
						|
 | 
						|
      var idx = 1, len = _.size(m);
 | 
						|
 | 
						|
      switch (mode) {
 | 
						|
        case 'properties':
 | 
						|
          idx = 0;
 | 
						|
          break;
 | 
						|
        case 'edit':
 | 
						|
        case 'create':
 | 
						|
        case 'control':
 | 
						|
          idx = 1;
 | 
						|
          break;
 | 
						|
        case 'cell':
 | 
						|
          idx = 2;
 | 
						|
          break;
 | 
						|
        default:
 | 
						|
          idx = 0;
 | 
						|
          break;
 | 
						|
      }
 | 
						|
 | 
						|
      return m[idx > len ? 0 : idx];
 | 
						|
    }
 | 
						|
    return type;
 | 
						|
  }
 | 
						|
 | 
						|
 | 
						|
  var BackformControlInit = Backform.Control.prototype.initialize,
 | 
						|
      BackformControlRemove = Backform.Control.prototype.remove;
 | 
						|
 | 
						|
  // Override the Backform.Control to allow to track changes in dependencies,
 | 
						|
  // and rerender the View element
 | 
						|
  _.extend(Backform.Control.prototype, {
 | 
						|
 | 
						|
    defaults: _.extend(Backform.Control.prototype.defaults, {helpMessage: null}),
 | 
						|
 | 
						|
    initialize: function() {
 | 
						|
      BackformControlInit.apply(this, arguments);
 | 
						|
 | 
						|
      // Listen to the dependent fields in the model for any change
 | 
						|
      var deps = this.field.get('deps');
 | 
						|
      var self = this;
 | 
						|
 | 
						|
      if (deps && _.isArray(deps)) {
 | 
						|
        _.each(deps, function(d) {
 | 
						|
          attrArr = d.split('.');
 | 
						|
          name = attrArr.shift();
 | 
						|
          self.listenTo(self.model, "change:" + name, self.render);
 | 
						|
        });
 | 
						|
      }
 | 
						|
    },
 | 
						|
 | 
						|
    remove: function() {
 | 
						|
      // Remove the events for the dependent fields in the model
 | 
						|
      var self = this,
 | 
						|
          deps = self.field.get('deps');
 | 
						|
 | 
						|
      self.stopListening(self.model, "change:" + name, self.render);
 | 
						|
      self.stopListening(self.model.errorModel, "change:" + name, self.updateInvalid);
 | 
						|
 | 
						|
      if (deps && _.isArray(deps)) {
 | 
						|
        _.each(deps, function(d) {
 | 
						|
 | 
						|
          attrArr = d.split('.');
 | 
						|
          name = attrArr.shift();
 | 
						|
 | 
						|
          self.stopListening(self.model, "change:" + name, self.render);
 | 
						|
        });
 | 
						|
      }
 | 
						|
 | 
						|
      if (this.cleanup) {
 | 
						|
        this.cleanup.apply(this);
 | 
						|
      }
 | 
						|
 | 
						|
      if (BackformControlRemove) {
 | 
						|
        BackformControlRemove.apply(self, arguments);
 | 
						|
      } else {
 | 
						|
        Backbone.View.prototype.remove.apply(self, arguments);
 | 
						|
      }
 | 
						|
    },
 | 
						|
 | 
						|
    template: _.template([
 | 
						|
                  '<label class="<%=Backform.controlLabelClassName%>"><%=label%></label>',
 | 
						|
                  '<div class="<%=Backform.controlsClassName%>">',
 | 
						|
                  '  <span class="<%=Backform.controlClassName%> uneditable-input" <%=disabled ? "disabled" : ""%>>',
 | 
						|
                  '    <%=value%>',
 | 
						|
                  '  </span>',
 | 
						|
                  '</div>',
 | 
						|
                  '<% if (helpMessage && helpMessage.length) { %>',
 | 
						|
                  '  <span class="<%=Backform.helpMessageClassName%>"><%=helpMessage%></span>',
 | 
						|
                  '<% } %>',
 | 
						|
                  ].join("\n")),
 | 
						|
 | 
						|
    clearInvalid: function() {
 | 
						|
      this.$el.removeClass(Backform.errorClassName);
 | 
						|
      this.$el.find(".pgadmin-control-error-message").remove();
 | 
						|
      return this;
 | 
						|
    },
 | 
						|
 | 
						|
    updateInvalid: function() {
 | 
						|
      var self = this,
 | 
						|
          errorModel = this.model.errorModel;
 | 
						|
 | 
						|
      if (!(errorModel instanceof Backbone.Model)) return this;
 | 
						|
 | 
						|
      this.clearInvalid();
 | 
						|
 | 
						|
      /*
 | 
						|
      * Find input which have name attribute.
 | 
						|
      */
 | 
						|
      this.$el.find(':input[name]').not('button').each(function(ix, el) {
 | 
						|
        var attrArr = $(el).attr('name').split('.'),
 | 
						|
        name = attrArr.shift(),
 | 
						|
        path = attrArr.join('.'),
 | 
						|
        error = self.keyPathAccessor(errorModel.toJSON(), $(el).attr('name'));
 | 
						|
 | 
						|
      if (_.isEmpty(error)) return;
 | 
						|
 | 
						|
      self.$el.addClass(Backform.errorClassName).append(
 | 
						|
        $("<div></div>").addClass('pgadmin-control-error-message col-xs-offset-4 col-xs-8 help-block').text(error)
 | 
						|
        );
 | 
						|
      });
 | 
						|
    },
 | 
						|
 | 
						|
    /*
 | 
						|
     * Overriding the render function of the control to allow us to eval the
 | 
						|
     * values properly.
 | 
						|
     */
 | 
						|
    render: function() {
 | 
						|
      var field = _.defaults(this.field.toJSON(), this.defaults),
 | 
						|
          attributes = this.model.toJSON(),
 | 
						|
          attrArr = field.name.split('.'),
 | 
						|
          name = attrArr.shift(),
 | 
						|
          path = attrArr.join('.'),
 | 
						|
          rawValue = this.keyPathAccessor(attributes[name], path),
 | 
						|
          data = _.extend(field, {
 | 
						|
            rawValue: rawValue,
 | 
						|
            value: this.formatter.fromRaw(rawValue, this.model),
 | 
						|
            attributes: attributes,
 | 
						|
            formatter: this.formatter
 | 
						|
          }),
 | 
						|
          evalF = function(f, d, m) {
 | 
						|
            return (_.isFunction(f) ? !!f.apply(d, [m]) : !!f);
 | 
						|
          };
 | 
						|
 | 
						|
      // Evaluate the disabled, visible, and required option
 | 
						|
      _.extend(data, {
 | 
						|
        disabled: evalF(data.disabled, data, this.model),
 | 
						|
        visible:  evalF(data.visible, data, this.model),
 | 
						|
        required: evalF(data.required, data, this.model)
 | 
						|
      });
 | 
						|
 | 
						|
      // Clean up first
 | 
						|
      this.$el.removeClass(Backform.hiddenClassname);
 | 
						|
 | 
						|
      if (!data.visible)
 | 
						|
        this.$el.addClass(Backform.hiddenClassname);
 | 
						|
 | 
						|
      this.$el.html(this.template(data)).addClass(field.name);
 | 
						|
      this.updateInvalid();
 | 
						|
 | 
						|
      return this;
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  /*
 | 
						|
   * Override the input control events in order to reslove the issue related to
 | 
						|
   * not updating the value sometimes in the input control.
 | 
						|
   */
 | 
						|
  _.extend(
 | 
						|
    Backform.InputControl.prototype, {
 | 
						|
      events: {
 | 
						|
        "change input": "onChange",
 | 
						|
        "blur input": "onChange",
 | 
						|
        "keyup input": "onKeyUp",
 | 
						|
        "focus input": "clearInvalid"
 | 
						|
      },
 | 
						|
      onKeyUp: function(ev) {
 | 
						|
        if (this.key_timeout) {
 | 
						|
          clearTimeout(this.key_timeout);
 | 
						|
        }
 | 
						|
 | 
						|
        this.keyup_timeout = setTimeout(function() {
 | 
						|
          this.onChange(ev);
 | 
						|
        }.bind(this), 400);
 | 
						|
      }
 | 
						|
    });
 | 
						|
 | 
						|
  /*
 | 
						|
   * Override the textarea control events in order to resolve the issue related
 | 
						|
   * to not updating the value in model on certain browsers in few situations
 | 
						|
   * like copy/paste, deletion using backspace.
 | 
						|
   *
 | 
						|
   * Reference:
 | 
						|
   * http://stackoverflow.com/questions/11338592/how-can-i-bind-to-the-change-event-of-a-textarea-in-jquery
 | 
						|
   */
 | 
						|
  _.extend(
 | 
						|
    Backform.TextareaControl.prototype, {
 | 
						|
      defaults: _.extend(
 | 
						|
        Backform.TextareaControl.prototype.defaults, {rows: 5, helpMessage: null}
 | 
						|
      ),
 | 
						|
      events : {
 | 
						|
        "change textarea": "onChange",
 | 
						|
        "keyup textarea": "onKeyUp",
 | 
						|
        "paste textarea": "onChange",
 | 
						|
        "selectionchange textarea": "onChange",
 | 
						|
        "focus textarea": "clearInvalid"
 | 
						|
      },
 | 
						|
      template: _.template([
 | 
						|
        '<label class="<%=Backform.controlLabelClassName%>"><%=label%></label>',
 | 
						|
        '<div class="<%=Backform.controlsClassName%>">',
 | 
						|
        '  <textarea ',
 | 
						|
        '    class="<%=Backform.controlClassName%> <%=extraClasses.join(\' \')%>" name="<%=name%>"',
 | 
						|
        '    maxlength="<%=maxlength%>" placeholder="<%-placeholder%>" <%=disabled ? "disabled" : ""%>',
 | 
						|
        '    rows=<%=rows ? rows : ""%>',
 | 
						|
        '    <%=required ? "required" : ""%>><%-value%></textarea>',
 | 
						|
        '  <% if (helpMessage && helpMessage.length) { %>',
 | 
						|
        '    <span class="<%=Backform.helpMessageClassName%>"><%=helpMessage%></span>',
 | 
						|
        '  <% } %>',
 | 
						|
        '</div>'
 | 
						|
      ].join("\n")),
 | 
						|
      onKeyUp: function(ev) {
 | 
						|
        if (this.key_timeout) {
 | 
						|
          clearTimeout(this.key_timeout);
 | 
						|
        }
 | 
						|
 | 
						|
        this.keyup_timeout = setTimeout(function() {
 | 
						|
          this.onChange(ev);
 | 
						|
        }.bind(this), 400);
 | 
						|
      }
 | 
						|
    });
 | 
						|
 | 
						|
  /*
 | 
						|
   * Overriding the render function of the select control to allow us to use
 | 
						|
   * options as function, which should return array in the format of
 | 
						|
   * (label, value) pair.
 | 
						|
   */
 | 
						|
  Backform.SelectControl.prototype.render = function() {
 | 
						|
    var field = _.defaults(this.field.toJSON(), this.defaults),
 | 
						|
      attributes = this.model.toJSON(),
 | 
						|
      attrArr = field.name.split('.'),
 | 
						|
      name = attrArr.shift(),
 | 
						|
      path = attrArr.join('.'),
 | 
						|
      rawValue = this.keyPathAccessor(attributes[name], path),
 | 
						|
      data = _.extend(field, {
 | 
						|
        rawValue: rawValue,
 | 
						|
        value: this.formatter.fromRaw(rawValue, this.model),
 | 
						|
        attributes: attributes,
 | 
						|
        formatter: this.formatter
 | 
						|
      }),
 | 
						|
      evalF = function(f, d, m) {
 | 
						|
        return (_.isFunction(f) ? !!f.apply(d, [m]) : !!f);
 | 
						|
      };
 | 
						|
 | 
						|
    // Evaluate the disabled, visible, and required option
 | 
						|
    _.extend(data, {
 | 
						|
      disabled: evalF(data.disabled, data, this.model),
 | 
						|
      visible:  evalF(data.visible, data, this.model),
 | 
						|
      required: evalF(data.required, data, this.model)
 | 
						|
    });
 | 
						|
    // Evaluation the options
 | 
						|
    if (_.isFunction(data.options)) {
 | 
						|
      try {
 | 
						|
        data.options = data.options(this)
 | 
						|
      } catch(e) {
 | 
						|
        // Do nothing
 | 
						|
        data.options = []
 | 
						|
        this.model.trigger('pgadmin-view:transform:error', this.model, this.field, e);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Clean up first
 | 
						|
    this.$el.removeClass(Backform.hiddenClassname);
 | 
						|
 | 
						|
    if (!data.visible)
 | 
						|
      this.$el.addClass(Backform.hiddenClassname);
 | 
						|
 | 
						|
    this.$el.html(this.template(data)).addClass(field.name);
 | 
						|
    this.updateInvalid();
 | 
						|
 | 
						|
    return this;
 | 
						|
  };
 | 
						|
  _.extend(Backform.SelectControl.prototype.defaults, {helpMessage: null});
 | 
						|
 | 
						|
  var ReadonlyOptionControl = Backform.ReadonlyOptionControl = Backform.SelectControl.extend({
 | 
						|
    template: _.template([
 | 
						|
      '<label class="<%=Backform.controlLabelClassName%>"><%=label%></label>',
 | 
						|
      '<div class="<%=Backform.controlsClassName%>">',
 | 
						|
      '<% for (var i=0; i < options.length; i++) { %>',
 | 
						|
      ' <% var option = options[i]; %>',
 | 
						|
      ' <% if (option.value === rawValue) { %>',
 | 
						|
      ' <span class="<%=Backform.controlClassName%> uneditable-input" disabled><%-option.label%></span>',
 | 
						|
      ' <% } %>',
 | 
						|
      '<% } %>',
 | 
						|
      '<% if (helpMessage && helpMessage.length) { %>',
 | 
						|
      '  <span class="<%=Backform.helpMessageClassName%>"><%=helpMessage%></span>',
 | 
						|
      '<% } %>',
 | 
						|
      '</div>'
 | 
						|
    ].join("\n")),
 | 
						|
    events: {},
 | 
						|
    getValueFromDOM: function() {
 | 
						|
      return this.formatter.toRaw(this.$el.find("span").text(), this.model);
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  /*
 | 
						|
   * Override the function 'updateInvalid' of the radio control to resolve an
 | 
						|
   * issue, which will not render the error block multiple times for each
 | 
						|
   * options.
 | 
						|
   */
 | 
						|
  _.extend(
 | 
						|
    Backform.RadioControl.prototype, {
 | 
						|
      updateInvalid: function() {
 | 
						|
        var self = this,
 | 
						|
            errorModel = this.model.errorModel;
 | 
						|
 | 
						|
        if (!(errorModel instanceof Backbone.Model)) return this;
 | 
						|
 | 
						|
        this.clearInvalid();
 | 
						|
 | 
						|
        /*
 | 
						|
         * Find input which have name attribute.
 | 
						|
         */
 | 
						|
        this.$el.find(':input[name]').not('button').each(function(ix, el) {
 | 
						|
          var attrArr = $(el).attr('name').split('.'),
 | 
						|
              name = attrArr.shift(),
 | 
						|
              path = attrArr.join('.'),
 | 
						|
              error = self.keyPathAccessor(
 | 
						|
                errorModel.toJSON(), $(el).attr('name')
 | 
						|
              );
 | 
						|
 | 
						|
          if (_.isEmpty(error)) return;
 | 
						|
 | 
						|
          self.$el.addClass(Backform.errorClassName).find(
 | 
						|
            '[type="radio"]'
 | 
						|
          ).append(
 | 
						|
          $("<div></div>").addClass(
 | 
						|
            'pgadmin-control-error-message col-xs-offset-4 col-xs-8 col-xs-8 help-block'
 | 
						|
          ).text(error));
 | 
						|
        });
 | 
						|
      }
 | 
						|
    });
 | 
						|
 | 
						|
  // Requires the Bootstrap Switch to work.
 | 
						|
  var SwitchControl = Backform.SwitchControl = Backform.InputControl.extend({
 | 
						|
    defaults: {
 | 
						|
      label: "",
 | 
						|
      options: {
 | 
						|
        onText: 'Yes',
 | 
						|
        offText: 'No',
 | 
						|
        onColor: 'success',
 | 
						|
        offColor: 'primary',
 | 
						|
        size: 'small'
 | 
						|
      },
 | 
						|
      extraClasses: [],
 | 
						|
      helpMessage: null
 | 
						|
    },
 | 
						|
    template: _.template([
 | 
						|
      '<label class="<%=Backform.controlLabelClassName%>"><%=label%></label>',
 | 
						|
      '<div class="<%=Backform.controlsClassName%>">',
 | 
						|
      '  <div class="checkbox">',
 | 
						|
      '    <label>',
 | 
						|
      '      <input type="checkbox" class="<%=extraClasses.join(\' \')%>" name="<%=name%>" <%=value ? "checked=\'checked\'" : ""%> <%=disabled ? "disabled" : ""%> <%=required ? "required" : ""%> />',
 | 
						|
      '    </label>',
 | 
						|
      '  </div>',
 | 
						|
      '</div>',
 | 
						|
      '<% if (helpMessage && helpMessage.length) { %>',
 | 
						|
      '  <span class="<%=Backform.helpMessageClassName%>"><%=helpMessage%></span>',
 | 
						|
      '<% } %>'
 | 
						|
    ].join("\n")),
 | 
						|
    getValueFromDOM: function() {
 | 
						|
      return this.formatter.toRaw(
 | 
						|
          this.$input.prop('checked'),
 | 
						|
          this.model
 | 
						|
          );
 | 
						|
    },
 | 
						|
    events: {'switchChange.bootstrapSwitch': 'onChange'},
 | 
						|
    render: function() {
 | 
						|
      var field = _.defaults(this.field.toJSON(), this.defaults),
 | 
						|
          attributes = this.model.toJSON(),
 | 
						|
          attrArr = field.name.split('.'),
 | 
						|
          name = attrArr.shift(),
 | 
						|
          path = attrArr.join('.'),
 | 
						|
          rawValue = this.keyPathAccessor(attributes[name], path),
 | 
						|
          evalF = function(f, d, m) {
 | 
						|
            return (_.isFunction(f) ? !!f.apply(d, [m]) : !!f);
 | 
						|
            },
 | 
						|
          options = _.defaults({
 | 
						|
              disabled: evalF(field.disabled, field, this.model)
 | 
						|
            }, this.field.get('options'), this.defaults.options,
 | 
						|
            $.fn.bootstrapSwitch.defaults);
 | 
						|
 | 
						|
      Backform.InputControl.prototype.render.apply(this, arguments);
 | 
						|
      this.$input = this.$el.find("input[type=checkbox]").first();
 | 
						|
 | 
						|
      //Check & set additional properties
 | 
						|
      this.$input.bootstrapSwitch(
 | 
						|
          _.extend(options, {'state': rawValue})
 | 
						|
          );
 | 
						|
 | 
						|
      return this;
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  // Backform Dialog view (in bootstrap tabbular form)
 | 
						|
  // A collection of field models.
 | 
						|
  var Dialog = Backform.Dialog = Backform.Form.extend({
 | 
						|
    /* Array of objects having attributes [label, fields] */
 | 
						|
    schema: undefined,
 | 
						|
    tagName: "form",
 | 
						|
    legend: true,
 | 
						|
    className: function() {
 | 
						|
      return 'col-sm-12 col-md-12 col-lg-12 col-xs-12';
 | 
						|
    },
 | 
						|
    tabPanelClassName: function() {
 | 
						|
      return Backform.tabClassName;
 | 
						|
    },
 | 
						|
    tabIndex: 0,
 | 
						|
    initialize: function(opts) {
 | 
						|
      var s = opts.schema;
 | 
						|
      if (s && _.isArray(s)) {
 | 
						|
        this.schema = _.each(s, function(o) {
 | 
						|
          if (o.fields && !(o.fields instanceof Backbone.Collection))
 | 
						|
            o.fields = new Backform.Fields(o.fields);
 | 
						|
          o.cId = o.cId || _.uniqueId('pgC_');
 | 
						|
          o.hId = o.hId || _.uniqueId('pgH_');
 | 
						|
          o.disabled = o.disabled || false;
 | 
						|
          o.legend = opts.legend;
 | 
						|
        });
 | 
						|
        if (opts.tabPanelClassName && _.isFunction(opts.tabPanelClassName)) {
 | 
						|
          this.tabPanelClassName = opts.tabPanelClassName;
 | 
						|
        }
 | 
						|
      }
 | 
						|
      this.model.errorModel = opts.errorModel || this.model.errorModel || new Backbone.Model();
 | 
						|
      this.controls = [];
 | 
						|
    },
 | 
						|
    template: {
 | 
						|
      'header': _.template([
 | 
						|
        '<li role="presentation" <%=disabled ? "disabled" : ""%>>',
 | 
						|
        ' <a data-toggle="tab" data-tab-index="<%=tabIndex%>" href="#<%=cId%>"',
 | 
						|
        '  id="<%=hId%>" aria-controls="<%=cId%>">',
 | 
						|
        '<%=label%></a></li>'].join(" ")),
 | 
						|
      'panel': _.template(
 | 
						|
        '<div role="tabpanel" class="tab-pane <%=label%> col-sm-12 col-md-12 col-lg-12 col-xs-12 fade" id="<%=cId%>" aria-labelledby="<%=hId%>"></div>'
 | 
						|
      )},
 | 
						|
    render: function() {
 | 
						|
      this.cleanup();
 | 
						|
 | 
						|
      var c = this.$el
 | 
						|
        .children().first().children('.active')
 | 
						|
        .first().attr('id'),
 | 
						|
        m = this.model,
 | 
						|
        controls = this.controls,
 | 
						|
        tmpls = this.template,
 | 
						|
        self = this,
 | 
						|
        idx=(this.tabIndex * 100),
 | 
						|
        evalF = function(f, d, m) {
 | 
						|
            return (_.isFunction(f) ? !!f.apply(d, [m]) : !!f);
 | 
						|
          };
 | 
						|
 | 
						|
      this.$el
 | 
						|
          .empty()
 | 
						|
          .attr('role', 'tabpanel')
 | 
						|
          .attr('class', _.result(this, 'tabPanelClassName'));
 | 
						|
      m.panelEl = this.$el;
 | 
						|
 | 
						|
      var tabHead = $('<ul class="nav nav-tabs" role="tablist"></ul>')
 | 
						|
        .appendTo(this.$el);
 | 
						|
      var tabContent = $('<ul class="tab-content col-sm-12 col-md-12 col-lg-12 col-xs-12"></ul>')
 | 
						|
        .appendTo(this.$el);
 | 
						|
 | 
						|
      _.each(this.schema, function(o) {
 | 
						|
        idx++;
 | 
						|
        if (!o.version_compatible || !evalF(o.visible, o, m)) {
 | 
						|
          return;
 | 
						|
        }
 | 
						|
        var el = $((tmpls['panel'])(_.extend(o, {'tabIndex': idx})))
 | 
						|
              .appendTo(tabContent)
 | 
						|
              .removeClass('collapse').addClass('collapse'),
 | 
						|
            h = $((tmpls['header'])(o)).appendTo(tabHead);
 | 
						|
 | 
						|
        o.fields.each(function(f) {
 | 
						|
          var cntr = new (f.get("control")) ({
 | 
						|
            field: f,
 | 
						|
            model: m,
 | 
						|
            dialog: self,
 | 
						|
            tabIndex: idx
 | 
						|
          });
 | 
						|
          el.append(cntr.render().$el);
 | 
						|
          controls.push(cntr);
 | 
						|
        });
 | 
						|
 | 
						|
        tabHead.find('a[data-toggle="tab"]').off(
 | 
						|
          'shown.bs.tab'
 | 
						|
        ).off('hidden.bs.tab').on(
 | 
						|
          'hidden.bs.tab', function() {
 | 
						|
            self.hidden_tab = $(this).data('tabIndex');
 | 
						|
            }).on('shown.bs.tab', function() {
 | 
						|
              var self = this;
 | 
						|
              self.shown_tab = $(self).data('tabIndex');
 | 
						|
              m.trigger('pg-property-tab-changed', {
 | 
						|
                'model': m, 'shown': self.shown_tab, 'hidden': self.hidden_tab,
 | 
						|
                'tab': self
 | 
						|
              });
 | 
						|
            });
 | 
						|
      });
 | 
						|
 | 
						|
      var makeActive = tabHead.find('[id="' + c + '"]').first();
 | 
						|
      if (makeActive.length == 1) {
 | 
						|
        makeActive.parent().addClass('active');
 | 
						|
        tabContent.find('#' + makeActive.attr("aria-controls"))
 | 
						|
          .addClass('in active');
 | 
						|
      } else {
 | 
						|
        tabHead.find('[role="presentation"]').first().addClass('active');
 | 
						|
        tabContent.find('[role="tabpanel"]').first().addClass('in active');
 | 
						|
      }
 | 
						|
 | 
						|
      return this;
 | 
						|
    },
 | 
						|
    remove: function(opts) {
 | 
						|
      if (opts && opts.data) {
 | 
						|
        if (this.model) {
 | 
						|
          if (this.model.reset) {
 | 
						|
            this.model.reset();
 | 
						|
          }
 | 
						|
          this.model.clear({silent: true});
 | 
						|
          delete (this.model);
 | 
						|
        }
 | 
						|
        if (this.errorModel) {
 | 
						|
          this.errorModel.clear({silent: true});
 | 
						|
          delete (this.errorModel);
 | 
						|
        }
 | 
						|
      }
 | 
						|
      this.cleanup();
 | 
						|
      Backform.Form.prototype.remove.apply(this, arguments);
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  var Fieldset = Backform.Fieldset = Backform.Dialog.extend({
 | 
						|
    className: function() {
 | 
						|
      return 'set-group col-xs-12';
 | 
						|
    },
 | 
						|
    tabPanelClassName: function() {
 | 
						|
      return Backform.tabClassName;
 | 
						|
    },
 | 
						|
    fieldsetClass:  Backform.setGroupClassName,
 | 
						|
    legendClass: 'badge',
 | 
						|
    contentClass: Backform.setGroupContentClassName + ' collapse in',
 | 
						|
    template: {
 | 
						|
      'header': _.template([
 | 
						|
        '<fieldset class="<%=fieldsetClass%>" <%=disabled ? "disabled" : ""%>>',
 | 
						|
        ' <% if (legend != false) { %>',
 | 
						|
        '  <legend class="<%=legendClass%>" <%=collapse ? "data-toggle=\'collapse\'" : ""%> data-target="#<%=cId%>"><%=collapse ? "<span class=\'caret\'></span>" : "" %><%=label%></legend>',
 | 
						|
        ' <% } %>',
 | 
						|
        '</fieldset>'
 | 
						|
      ].join("\n")),
 | 
						|
      'content': _.template(
 | 
						|
        '  <div id="<%= cId %>" class="<%=contentClass%>"></div>'
 | 
						|
    )},
 | 
						|
    collapse: true,
 | 
						|
    render: function() {
 | 
						|
      this.cleanup();
 | 
						|
 | 
						|
      var m = this.model,
 | 
						|
          $el = this.$el,
 | 
						|
          tmpl = this.template,
 | 
						|
          controls = this.controls,
 | 
						|
          data = {
 | 
						|
            'className': _.result(this, 'className'),
 | 
						|
            'fieldsetClass': _.result(this, 'fieldsetClass'),
 | 
						|
            'legendClass': _.result(this, 'legendClass'),
 | 
						|
            'contentClass': _.result(this, 'contentClass'),
 | 
						|
            'collapse': _.result(this, 'collapse')
 | 
						|
          },
 | 
						|
          idx=(this.tabIndex * 100),
 | 
						|
          evalF = function(f, d, m) {
 | 
						|
            return (_.isFunction(f) ? !!f.apply(d, [m]) : !!f);
 | 
						|
          };
 | 
						|
 | 
						|
      this.$el.empty();
 | 
						|
 | 
						|
      _.each(this.schema, function(o) {
 | 
						|
        idx++;
 | 
						|
        if (!o.version_compatible || !evalF(o.visible, o, m)) {
 | 
						|
          return;
 | 
						|
        }
 | 
						|
 | 
						|
        if (!o.fields)
 | 
						|
          return;
 | 
						|
 | 
						|
        var d = _.extend({}, data, o),
 | 
						|
            h = $((tmpl['header'])(d)).appendTo($el),
 | 
						|
            el = $((tmpl['content'])(d)).appendTo(h);
 | 
						|
 | 
						|
        o.fields.each(function(f) {
 | 
						|
          var cntr = new (f.get("control")) ({
 | 
						|
            field: f,
 | 
						|
            model: m,
 | 
						|
            tabIndex: idx
 | 
						|
          });
 | 
						|
          el.append(cntr.render().$el);
 | 
						|
          controls.push(cntr);
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      return this;
 | 
						|
    },
 | 
						|
    getValueFromDOM: function() {
 | 
						|
      return "";
 | 
						|
    },
 | 
						|
    events: {}
 | 
						|
  });
 | 
						|
 | 
						|
  var generateGridColumnsFromModel = Backform.generateGridColumnsFromModel =
 | 
						|
    function(node_info, m, type, cols, node) {
 | 
						|
      var groups = Backform.generateViewSchema(node_info, m, type, node, true, true),
 | 
						|
          schema = [],
 | 
						|
          columns = [],
 | 
						|
          func,
 | 
						|
          idx = 0;
 | 
						|
 | 
						|
      // Create another array if cols is of type object & store its keys in that array,
 | 
						|
      // If cols is object then chances that we have custom width class attached with in.
 | 
						|
      if (_.isNull(cols) || _.isUndefined(cols)) {
 | 
						|
        func = function(f) {
 | 
						|
          f.cell_priority = idx;
 | 
						|
          idx = idx + 1;
 | 
						|
 | 
						|
          // We can also provide custom header cell class in schema itself,
 | 
						|
          // But we will give priority to extraClass attached in cols
 | 
						|
          // If headerCell property is already set by cols then skip extraClass property from schema
 | 
						|
          if (!(f.headerCell) && f.cellHeaderClasses) {
 | 
						|
            f.headerCell = Backgrid.Extension.CustomHeaderCell;
 | 
						|
          }
 | 
						|
        };
 | 
						|
      } else if (_.isArray(cols)) {
 | 
						|
        func = function(f) {
 | 
						|
          f.cell_priority = _.indexOf(cols, f.name);
 | 
						|
 | 
						|
          // We can also provide custom header cell class in schema itself,
 | 
						|
          // But we will give priority to extraClass attached in cols
 | 
						|
          // If headerCell property is already set by cols then skip extraClass property from schema
 | 
						|
          if ((!f.headerCell) && f.cellHeaderClasses) {
 | 
						|
            f.headerCell = Backgrid.Extension.CustomHeaderCell;
 | 
						|
          }
 | 
						|
        };
 | 
						|
      } else if(_.isObject(cols)) {
 | 
						|
        var tblCols = Object.keys(cols);
 | 
						|
        func = function(f) {
 | 
						|
          var val = (f.name in cols) && cols[f.name];
 | 
						|
 | 
						|
          if (_.isNull(val) || _.isUndefined(val)) {
 | 
						|
            f.cell_priority = -1;
 | 
						|
            return;
 | 
						|
          }
 | 
						|
          if (_.isObject(val)) {
 | 
						|
            if ('index' in val) {
 | 
						|
              f.cell_priority = val['index'];
 | 
						|
              idx = (idx > val['index']) ? idx + 1 : val['index'];
 | 
						|
            } else {
 | 
						|
              var i = _.indexOf(tblCols, f.name);
 | 
						|
              f.cell_priority = idx = ((i > idx) ? i : idx);
 | 
						|
              idx = idx + 1;
 | 
						|
            }
 | 
						|
 | 
						|
            // We can also provide custom header cell class in schema itself,
 | 
						|
            // But we will give priority to extraClass attached in cols
 | 
						|
            // If headerCell property is already set by cols then skip extraClass property from schema
 | 
						|
            if (!f.headerCell) {
 | 
						|
              if (f.cellHeaderClasses) {
 | 
						|
                f.headerCell = Backgrid.Extension.CustomHeaderCell;
 | 
						|
              }
 | 
						|
              if ('class' in val && _.isString(val['class'])) {
 | 
						|
                f.headerCell = Backgrid.Extension.CustomHeaderCell;
 | 
						|
                f.cellHeaderClasses = (f.cellHeaderClasses || '') + ' ' + val['class'];
 | 
						|
              }
 | 
						|
            }
 | 
						|
          }
 | 
						|
          if (_.isString(val)) {
 | 
						|
            var i = _.indexOf(tblCols, f.name);
 | 
						|
 | 
						|
            f.cell_priority = idx = ((i > idx) ? i : idx);
 | 
						|
            idx = idx + 1;
 | 
						|
 | 
						|
            if (!f.headerCell) {
 | 
						|
              f.headerCell = Backgrid.Extension.CustomHeaderCell;
 | 
						|
            }
 | 
						|
            f.cellHeaderClasses = (f.cellHeaderClasses || '') + ' ' + val;
 | 
						|
          }
 | 
						|
        };
 | 
						|
      }
 | 
						|
 | 
						|
      // Prepare columns for backgrid
 | 
						|
      _.each(groups, function(group, key) {
 | 
						|
        _.each(group.fields, function(f) {
 | 
						|
          if (!f.cell) {
 | 
						|
            return;
 | 
						|
          }
 | 
						|
          // Check custom property in cols & if it is present then attach it to current cell
 | 
						|
          func(f);
 | 
						|
          if (f.cell_priority != -1) {
 | 
						|
            columns.push(f);
 | 
						|
          }
 | 
						|
        });
 | 
						|
        schema.push(group);
 | 
						|
      });
 | 
						|
      return {
 | 
						|
        'columns': _.sortBy(columns, function(c) {
 | 
						|
          return c.cell_priority;
 | 
						|
        }),
 | 
						|
        'schema': schema
 | 
						|
      };
 | 
						|
    };
 | 
						|
 | 
						|
  var UniqueColCollectionControl = Backform.UniqueColCollectionControl = Backform.Control.extend({
 | 
						|
    initialize: function() {
 | 
						|
        Backform.Control.prototype.initialize.apply(this, arguments);
 | 
						|
 | 
						|
        var uniqueCol = this.field.get('uniqueCol') || [],
 | 
						|
            m = this.field.get('model'),
 | 
						|
            schema = m.prototype.schema || m.__super__.schema,
 | 
						|
            columns = [],
 | 
						|
            self = this;
 | 
						|
 | 
						|
        _.each(schema, function(s) {
 | 
						|
          columns.push(s.id);
 | 
						|
        });
 | 
						|
 | 
						|
        // Check if unique columns provided are also in model attributes.
 | 
						|
        if (uniqueCol.length > _.intersection(columns, uniqueCol).length) {
 | 
						|
            errorMsg = "Developer: Unique columns [ "+_.difference(uniqueCol, columns)+" ] not found in collection model [ " + columns +" ]."
 | 
						|
            alert (errorMsg);
 | 
						|
        }
 | 
						|
 | 
						|
        var collection = self.collection = self.model.get(self.field.get('name'));
 | 
						|
 | 
						|
        if (!collection) {
 | 
						|
          collection = self.collection = new (pgAdmin.Browser.Node.Collection)(
 | 
						|
              null,
 | 
						|
              {
 | 
						|
                model: self.field.get('model'),
 | 
						|
                silent: true,
 | 
						|
                handler: self.model.handler || self.model,
 | 
						|
                top: self.model.top || self.model,
 | 
						|
                attrName: self.field.get('name')
 | 
						|
              });
 | 
						|
          self.model.set(self.field.get('name'), collection, {silent: true});
 | 
						|
        }
 | 
						|
 | 
						|
        if (this.field.get('version_compatible')) {
 | 
						|
          self.listenTo(collection, "add", self.collectionChanged);
 | 
						|
          self.listenTo(collection, "change", self.collectionChanged);
 | 
						|
        }
 | 
						|
    },
 | 
						|
    cleanup: function() {
 | 
						|
      this.stopListening(this.collection, "change", this.collectionChanged);
 | 
						|
 | 
						|
      if (this.field.get('version_compatible')) {
 | 
						|
        this.stopListening(self.collection, "add", this.collectionChanged);
 | 
						|
        this.stopListening(self.collection, "change", this.collectionChanged);
 | 
						|
      }
 | 
						|
      if (this.grid) {
 | 
						|
        this.grid.remove();
 | 
						|
        delete this.grid;
 | 
						|
      }
 | 
						|
      this.$el.empty();
 | 
						|
    },
 | 
						|
    collectionChanged: function(newModel, coll, op) {
 | 
						|
      var uniqueCol = this.field.get('uniqueCol') || [],
 | 
						|
          uniqueChangedAttr = [],
 | 
						|
          self = this;
 | 
						|
      // Check if changed model attributes are also in unique columns. And then only check for uniqueness.
 | 
						|
      if (newModel.attributes) {
 | 
						|
        _.each(uniqueCol, function(col) {
 | 
						|
            if (_.has(newModel.attributes,col))
 | 
						|
            {
 | 
						|
               uniqueChangedAttr.push(col);
 | 
						|
            }
 | 
						|
        });
 | 
						|
        if(uniqueChangedAttr.length == 0) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
      } else {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
      var collection = this.model.get(this.field.get('name'));
 | 
						|
      this.stopListening(collection, "change", this.collectionChanged);
 | 
						|
      // Check if changed attribute's value of new/updated model also exist for another model in collection.
 | 
						|
      // If duplicate value exists then set the attribute's value of new/updated model to its previous values.
 | 
						|
      var m = undefined,
 | 
						|
          oldModel = undefined;
 | 
						|
      collection.each(function(model) {
 | 
						|
        if (newModel != model) {
 | 
						|
          var duplicateAttrValues = []
 | 
						|
          _.each(uniqueCol, function(attr) {
 | 
						|
            attrValue = newModel.get(attr);
 | 
						|
            if (!_.isUndefined(attrValue) && attrValue == model.get(attr)) {
 | 
						|
              duplicateAttrValues.push(attrValue)
 | 
						|
            }
 | 
						|
          });
 | 
						|
          if (duplicateAttrValues.length == uniqueCol.length) {
 | 
						|
            m = newModel;
 | 
						|
            // Keep reference of model to make it visible in dialog.
 | 
						|
            oldModel = model;
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
      if (m) {
 | 
						|
        if (op && op.add) {
 | 
						|
          // Remove duplicate model.
 | 
						|
          setTimeout(function() {
 | 
						|
            collection.remove(m);
 | 
						|
          }, 0);
 | 
						|
 | 
						|
        } else {
 | 
						|
          /*
 | 
						|
           * Set model value to its previous value as its new value is
 | 
						|
           * conflicting with another model value.
 | 
						|
           */
 | 
						|
 | 
						|
          m.set(uniqueChangedAttr[0], m.previous(uniqueChangedAttr[0]));
 | 
						|
        }
 | 
						|
        if (oldModel) {
 | 
						|
          var idx = collection.indexOf(oldModel);
 | 
						|
          if (idx > -1) {
 | 
						|
            var newRow = self.grid.body.rows[idx].$el;
 | 
						|
            newRow.addClass("new");
 | 
						|
            $(newRow).pgMakeVisible('backform-tab');
 | 
						|
            setTimeout(function() {
 | 
						|
              newRow.removeClass("new");
 | 
						|
              }, 3000);
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      this.listenTo(collection, "change", this.collectionChanged);
 | 
						|
    },
 | 
						|
    render: function() {
 | 
						|
      // Clean up existing elements
 | 
						|
 | 
						|
      this.undelegateEvents();
 | 
						|
      this.$el.empty();
 | 
						|
 | 
						|
      var field = _.defaults(this.field.toJSON(), this.defaults),
 | 
						|
          attributes = this.model.toJSON(),
 | 
						|
          attrArr = field.name.split('.'),
 | 
						|
          name = attrArr.shift(),
 | 
						|
          path = attrArr.join('.'),
 | 
						|
          rawValue = this.keyPathAccessor(attributes[name], path),
 | 
						|
          data = _.extend(field, {
 | 
						|
            rawValue: rawValue,
 | 
						|
            value: this.formatter.fromRaw(rawValue, this.model),
 | 
						|
            attributes: attributes,
 | 
						|
            formatter: this.formatter
 | 
						|
           }),
 | 
						|
          evalF = function(f, d, m) {
 | 
						|
            return (_.isFunction(f) ? !!f.apply(d, [m]) : !!f);
 | 
						|
          };
 | 
						|
 | 
						|
      // Evaluate the disabled, visible, required, canAdd, & canDelete option
 | 
						|
      _.extend(data, {
 | 
						|
        disabled: (field.version_compatible &&
 | 
						|
                   evalF.apply(this.field, [data.disabled, data, this.model])
 | 
						|
                   ),
 | 
						|
        visible:  evalF.apply(this.field, [data.visible, data, this.model]),
 | 
						|
        required: evalF.apply(this.field, [data.required, data, this.model]),
 | 
						|
        canAdd: (field.version_compatible &&
 | 
						|
          evalF.apply(this.field, [data.canAdd, data, this.model])
 | 
						|
          ),
 | 
						|
        canAddRow: data.canAddRow,
 | 
						|
        canDelete: evalF.apply(this.field, [data.canDelete, data, this.model]),
 | 
						|
        canEdit: evalF.apply(this.field, [data.canEdit, data, this.model])
 | 
						|
      });
 | 
						|
      _.extend(data, {add_label: "ADD"});
 | 
						|
 | 
						|
      // This control is not visible, we should remove it.
 | 
						|
      if (!data.visible) {
 | 
						|
        return this;
 | 
						|
      }
 | 
						|
 | 
						|
      this.control_data = _.clone(data);
 | 
						|
 | 
						|
      // Show Backgrid Control
 | 
						|
      grid = this.showGridControl(data);
 | 
						|
 | 
						|
      this.$el.html(grid).addClass(field.name);
 | 
						|
      this.updateInvalid();
 | 
						|
 | 
						|
      this.delegateEvents();
 | 
						|
      return this;
 | 
						|
    },
 | 
						|
    showGridControl: function(data) {
 | 
						|
      var gridHeader = _.template([
 | 
						|
          '<div class="subnode-header">',
 | 
						|
          '  <label class="control-label col-sm-4"><%-label%></label>',
 | 
						|
          '  <button class="btn-sm btn-default add" <%=canAdd ? "" : "disabled=\'disabled\'"%>><%-add_label%></buttton>',
 | 
						|
          '</div>'].join("\n")),
 | 
						|
        gridBody = $('<div class="pgadmin-control-group backgrid form-group col-xs-12 object subnode"></div>').append(
 | 
						|
            gridHeader(data)
 | 
						|
            );
 | 
						|
 | 
						|
      if (!(data.subnode)) {
 | 
						|
        return '';
 | 
						|
      }
 | 
						|
 | 
						|
      var subnode = data.subnode.schema ? data.subnode : data.subnode.prototype,
 | 
						|
          gridSchema = Backform.generateGridColumnsFromModel(
 | 
						|
            data.node_info, subnode, this.field.get('mode'), data.columns
 | 
						|
            ),
 | 
						|
          self = this;
 | 
						|
 | 
						|
      // Set visibility of Add button
 | 
						|
      if (data.mode == 'properties') {
 | 
						|
        $(gridBody).find("button.add").remove();
 | 
						|
      }
 | 
						|
 | 
						|
      // Insert Delete Cell into Grid
 | 
						|
      if (!data.disabled && data.canDelete) {
 | 
						|
          gridSchema.columns.unshift({
 | 
						|
            name: "pg-backform-delete", label: "",
 | 
						|
            cell: Backgrid.Extension.DeleteCell,
 | 
						|
            editable: false, cell_priority: -1,
 | 
						|
            canDeleteRow: data.canDeleteRow
 | 
						|
          });
 | 
						|
      }
 | 
						|
 | 
						|
      // Insert Edit Cell into Grid
 | 
						|
      if (data.disabled == false && data.canEdit) {
 | 
						|
          var editCell = Backgrid.Extension.ObjectCell.extend({
 | 
						|
            schema: gridSchema.schema
 | 
						|
          });
 | 
						|
 | 
						|
          gridSchema.columns.unshift({
 | 
						|
            name: "pg-backform-edit", label: "", cell : editCell,
 | 
						|
            cell_priority: -2, canEditRow: data.canEditRow
 | 
						|
          });
 | 
						|
      }
 | 
						|
 | 
						|
      var collection = this.model.get(data.name);
 | 
						|
 | 
						|
      var cellEditing = function(args){
 | 
						|
        var self = this,
 | 
						|
          cell = args[0];
 | 
						|
        // Search for any other rows which are open.
 | 
						|
        this.each(function(m){
 | 
						|
          // Check if row which we are about to close is not current row.
 | 
						|
          if (cell.model != m) {
 | 
						|
            var idx = self.indexOf(m);
 | 
						|
            if (idx > -1) {
 | 
						|
              var row = grid.body.rows[idx],
 | 
						|
                  editCell = row.$el.find(".subnode-edit-in-process").parent();
 | 
						|
              // Only close row if it's open.
 | 
						|
              if (editCell.length > 0){
 | 
						|
                var event = new Event('click');
 | 
						|
                editCell[0].dispatchEvent(event);
 | 
						|
              }
 | 
						|
            }
 | 
						|
          }
 | 
						|
        });
 | 
						|
      }
 | 
						|
      // Listen for any row which is about to enter in edit mode.
 | 
						|
      collection.on( "enteringEditMode", cellEditing, collection);
 | 
						|
 | 
						|
      // Initialize a new Grid instance
 | 
						|
      var grid = self.grid = new Backgrid.Grid({
 | 
						|
        columns: gridSchema.columns,
 | 
						|
        collection: collection,
 | 
						|
        className: "backgrid table-bordered"
 | 
						|
      });
 | 
						|
 | 
						|
      // Render subNode grid
 | 
						|
      subNodeGrid = grid.render().$el;
 | 
						|
 | 
						|
      // Combine Edit and Delete Cell
 | 
						|
      if (data.canDelete && data.canEdit) {
 | 
						|
        $(subNodeGrid).find("th.pg-backform-delete").remove();
 | 
						|
        $(subNodeGrid).find("th.pg-backform-edit").attr("colspan", "2");
 | 
						|
      }
 | 
						|
 | 
						|
      $dialog =  gridBody.append(subNodeGrid);
 | 
						|
 | 
						|
      // Add button callback
 | 
						|
      if (!(data.disabled || data.canAdd == false)) {
 | 
						|
        $dialog.find('button.add').first().click(function(e) {
 | 
						|
          e.preventDefault();
 | 
						|
          var canAddRow = _.isFunction(data.canAddRow) ?
 | 
						|
                            data.canAddRow.apply(self, [self.model]) : true;
 | 
						|
          if (canAddRow) {
 | 
						|
              // Close any existing expanded row before adding new one.
 | 
						|
              _.each(grid.body.rows, function(row){
 | 
						|
                var editCell = row.$el.find(".subnode-edit-in-process").parent();
 | 
						|
                // Only close row if it's open.
 | 
						|
                if (editCell.length > 0){
 | 
						|
                  var event = new Event('click');
 | 
						|
                  editCell[0].dispatchEvent(event);
 | 
						|
                }
 | 
						|
              });
 | 
						|
 | 
						|
              var allowMultipleEmptyRows = !!self.field.get('allowMultipleEmptyRows');
 | 
						|
 | 
						|
              // If allowMultipleEmptyRows is not set or is false then don't allow second new empty row.
 | 
						|
              // There should be only one empty row.
 | 
						|
              if (!allowMultipleEmptyRows && collection) {
 | 
						|
                var isEmpty = false;
 | 
						|
                collection.each(function(model) {
 | 
						|
                  var modelValues = [];
 | 
						|
                  _.each(model.attributes, function(val, key) {
 | 
						|
                    modelValues.push(val);
 | 
						|
                  })
 | 
						|
                  if(!_.some(modelValues, _.identity)) {
 | 
						|
                    isEmpty = true;
 | 
						|
                  }
 | 
						|
                });
 | 
						|
                if(isEmpty) {
 | 
						|
                  return false;
 | 
						|
                }
 | 
						|
              }
 | 
						|
 | 
						|
              $(grid.body.$el.find($("tr.new"))).removeClass("new")
 | 
						|
              var m = new (data.model) (null, {
 | 
						|
                silent: true,
 | 
						|
                handler: self.model.handler || self.model,
 | 
						|
                top: self.model.top || self.model,
 | 
						|
                node_info: self.model.node_info,
 | 
						|
                collection: collection
 | 
						|
              });
 | 
						|
              collection.add(m);
 | 
						|
 | 
						|
              var idx = collection.indexOf(m),
 | 
						|
                  newRow = grid.body.rows[idx].$el;
 | 
						|
 | 
						|
              newRow.addClass("new");
 | 
						|
              $(newRow).pgMakeVisible('backform-tab');
 | 
						|
 | 
						|
              return false;
 | 
						|
          }
 | 
						|
        });
 | 
						|
      }
 | 
						|
 | 
						|
      return $dialog;
 | 
						|
    },
 | 
						|
    clearInvalid: function() {
 | 
						|
      this.$el.removeClass("subnode-error");
 | 
						|
      this.$el.find(".pgadmin-control-error-message").remove();
 | 
						|
      return this;
 | 
						|
    },
 | 
						|
    updateInvalid: function() {
 | 
						|
      var self = this,
 | 
						|
      errorModel = this.model.errorModel;
 | 
						|
 | 
						|
      if (!(errorModel instanceof Backbone.Model)) return this;
 | 
						|
 | 
						|
      this.clearInvalid();
 | 
						|
 | 
						|
      this.$el.find('.subnode-body').each(function(ix, el) {
 | 
						|
        var error = self.keyPathAccessor(errorModel.toJSON(), self.field.get('name'));
 | 
						|
 | 
						|
        if (_.isEmpty(error)) return;
 | 
						|
 | 
						|
        self.$el.addClass("subnode-error").append(
 | 
						|
          $("<div></div>").addClass('pgadmin-control-error-message col-xs-offset-4 col-xs-8 help-block').text(error)
 | 
						|
          );
 | 
						|
      });
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  var SubNodeCollectionControl =  Backform.SubNodeCollectionControl = Backform.Control.extend({
 | 
						|
    render: function() {
 | 
						|
      var field = _.defaults(this.field.toJSON(), this.defaults),
 | 
						|
          attributes = this.model.toJSON(),
 | 
						|
          attrArr = field.name.split('.'),
 | 
						|
          name = attrArr.shift(),
 | 
						|
          path = attrArr.join('.'),
 | 
						|
          rawValue = this.keyPathAccessor(attributes[name], path),
 | 
						|
          data = _.extend(field, {
 | 
						|
            rawValue: rawValue,
 | 
						|
            value: this.formatter.fromRaw(rawValue, this.model),
 | 
						|
            attributes: attributes,
 | 
						|
            formatter: this.formatter
 | 
						|
           }),
 | 
						|
          evalF = function(f, d, m) {
 | 
						|
            return (_.isFunction(f) ? !!f.apply(d, [m]) : !!f);
 | 
						|
          };
 | 
						|
 | 
						|
      // Evaluate the disabled, visible, required, canAdd, cannEdit & canDelete option
 | 
						|
      _.extend(data, {
 | 
						|
        disabled: evalF(data.disabled, data, this.model),
 | 
						|
        visible:  evalF(data.visible, data, this.model),
 | 
						|
        required: evalF(data.required, data, this.model),
 | 
						|
        canAdd: evalF(data.canAdd, data, this.model),
 | 
						|
        canAddRow: data.canAddRow,
 | 
						|
        canEdit: evalF(data.canEdit, data, this.model),
 | 
						|
        canDelete: evalF(data.canDelete, data, this.model)
 | 
						|
      });
 | 
						|
      // Show Backgrid Control
 | 
						|
      grid = (data.subnode == undefined) ? "" : this.showGridControl(data);
 | 
						|
 | 
						|
      this.$el.html(grid).addClass(field.name);
 | 
						|
      this.updateInvalid();
 | 
						|
 | 
						|
      return this;
 | 
						|
    },
 | 
						|
    updateInvalid: function() {
 | 
						|
      var self = this;
 | 
						|
      var errorModel = this.model.errorModel;
 | 
						|
      if (!(errorModel instanceof Backbone.Model)) return this;
 | 
						|
 | 
						|
      this.clearInvalid();
 | 
						|
 | 
						|
      var attrArr = self.field.get('name').split('.'),
 | 
						|
        name = attrArr.shift(),
 | 
						|
        path = attrArr.join('.'),
 | 
						|
        error = self.keyPathAccessor(errorModel.toJSON(), path);
 | 
						|
 | 
						|
      if (_.isEmpty(error)) return;
 | 
						|
 | 
						|
      self.$el.addClass('subnode-error').append(
 | 
						|
        $("<div></div>").addClass('pgadmin-control-error-message col-xs-offset-4 col-xs-8 help-block').text(error)
 | 
						|
        );
 | 
						|
    },
 | 
						|
    clearInvalid: function() {
 | 
						|
      this.$el.removeClass('subnode-error');
 | 
						|
      this.$el.find(".pgadmin-control-error-message").remove();
 | 
						|
      return this;
 | 
						|
    },
 | 
						|
    showGridControl: function(data) {
 | 
						|
      var gridHeader = ["<div class='subnode-header'>",
 | 
						|
          "  <label class='control-label col-sm-4'>" + data.label + "</label>" ,
 | 
						|
          "  <button class='btn-sm btn-default add'>ADD</buttton>",
 | 
						|
          "</div>"].join("\n");
 | 
						|
        gridBody = $("<div class='pgadmin-control-group backgrid form-group col-xs-12 object subnode'></div>").append(gridHeader);
 | 
						|
 | 
						|
      var subnode = data.subnode.schema ? data.subnode : data.subnode.prototype,
 | 
						|
          gridSchema = Backform.generateGridColumnsFromModel(
 | 
						|
            data.node_info, subnode, this.field.get('mode'), data.columns, data.schema_node
 | 
						|
            ), self = this,
 | 
						|
          pgBrowser = window.pgAdmin.Browser;
 | 
						|
 | 
						|
      // Set visibility of Add button
 | 
						|
      if (data.disabled || data.canAdd == false) {
 | 
						|
        $(gridBody).find("button.add").remove();
 | 
						|
      }
 | 
						|
 | 
						|
      // Insert Delete Cell into Grid
 | 
						|
      if (data.disabled == false && data.canDelete) {
 | 
						|
          gridSchema.columns.unshift({
 | 
						|
            name: "pg-backform-delete", label: "",
 | 
						|
            cell: Backgrid.Extension.DeleteCell,
 | 
						|
            editable: false, cell_priority: -1,
 | 
						|
            canDeleteRow: data.canDeleteRow
 | 
						|
          });
 | 
						|
      }
 | 
						|
 | 
						|
      // Insert Edit Cell into Grid
 | 
						|
      if (data.disabled == false && data.canEdit) {
 | 
						|
          var editCell = Backgrid.Extension.ObjectCell.extend({
 | 
						|
            schema: gridSchema.schema
 | 
						|
          }),
 | 
						|
          canEdit = self.field.has('canEdit') &&
 | 
						|
          self.field.get('canEdit') || true;
 | 
						|
 | 
						|
          gridSchema.columns.unshift({
 | 
						|
            name: "pg-backform-edit", label: "", cell : editCell,
 | 
						|
            cell_priority: -2, editable: canEdit,
 | 
						|
            canEditRow: data.canEditRow
 | 
						|
          });
 | 
						|
      }
 | 
						|
 | 
						|
      var collection = self.model.get(data.name);
 | 
						|
 | 
						|
      if (!collection) {
 | 
						|
        collection = new (pgBrowser.Node.Collection)(null, {
 | 
						|
          handler: self.model.handler || self,
 | 
						|
          model: data.model,
 | 
						|
          silent: true
 | 
						|
        });
 | 
						|
        self.model.set(data.name, collection, {silent: true});
 | 
						|
      }
 | 
						|
 | 
						|
      var cellEditing = function(args){
 | 
						|
        var self = this,
 | 
						|
          cell = args[0];
 | 
						|
        // Search for any other rows which are open.
 | 
						|
        this.each(function(m){
 | 
						|
          // Check if row which we are about to close is not current row.
 | 
						|
          if (cell.model != m) {
 | 
						|
            var idx = self.indexOf(m);
 | 
						|
            if (idx > -1) {
 | 
						|
              var row = grid.body.rows[idx],
 | 
						|
                  editCell = row.$el.find(".subnode-edit-in-process").parent();
 | 
						|
              // Only close row if it's open.
 | 
						|
              if (editCell.length > 0){
 | 
						|
                var event = new Event('click');
 | 
						|
                editCell[0].dispatchEvent(event);
 | 
						|
              }
 | 
						|
            }
 | 
						|
          }
 | 
						|
        });
 | 
						|
      }
 | 
						|
      // Listen for any row which is about to enter in edit mode.
 | 
						|
      collection.on( "enteringEditMode", cellEditing, collection);
 | 
						|
 | 
						|
      // Initialize a new Grid instance
 | 
						|
      var grid = self.grid = new Backgrid.Grid({
 | 
						|
          columns: gridSchema.columns,
 | 
						|
          collection: collection,
 | 
						|
          className: "backgrid table-bordered"
 | 
						|
      });
 | 
						|
 | 
						|
      // Render subNode grid
 | 
						|
      subNodeGrid = grid.render().$el;
 | 
						|
 | 
						|
      // Combine Edit and Delete Cell
 | 
						|
      if (data.canDelete && data.canEdit) {
 | 
						|
        $(subNodeGrid).find("th.pg-backform-delete").remove();
 | 
						|
        $(subNodeGrid).find("th.pg-backform-edit").attr("colspan", "2");
 | 
						|
      }
 | 
						|
 | 
						|
      $dialog =  gridBody.append(subNodeGrid);
 | 
						|
 | 
						|
      // Add button callback
 | 
						|
      $dialog.find('button.add').click(function(e) {
 | 
						|
         e.preventDefault();
 | 
						|
        var canAddRow = _.isFunction(data.canAddRow) ?
 | 
						|
                            data.canAddRow.apply(self, [self.model]) : true;
 | 
						|
        if (canAddRow) {
 | 
						|
          // Close any existing expanded row before adding new one.
 | 
						|
          _.each(grid.body.rows, function(row){
 | 
						|
            var editCell = row.$el.find(".subnode-edit-in-process").parent();
 | 
						|
            // Only close row if it's open.
 | 
						|
            if (editCell.length > 0){
 | 
						|
              var event = new Event('click');
 | 
						|
              editCell[0].dispatchEvent(event);
 | 
						|
            }
 | 
						|
          });
 | 
						|
 | 
						|
          grid.insertRow({});
 | 
						|
 | 
						|
          var newRow = $(grid.body.rows[collection.length - 1].$el);
 | 
						|
          newRow.attr("class", "new").click(function(e) {
 | 
						|
            $(this).attr("class", "editable");
 | 
						|
          });
 | 
						|
          $(newRow).pgMakeVisible('backform-tab');
 | 
						|
          return false;
 | 
						|
        }
 | 
						|
      });
 | 
						|
 | 
						|
      return $dialog;
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  /*
 | 
						|
   * SQL Tab Control for showing the modified SQL for the node with the
 | 
						|
   * property 'hasSQL' is set to true.
 | 
						|
   *
 | 
						|
   * When the user clicks on the SQL tab, we will send the modified data to the
 | 
						|
   * server and fetch the SQL for it.
 | 
						|
   */
 | 
						|
  var SqlTabControl = Backform.SqlTabControl = Backform.Control.extend({
 | 
						|
    defaults: {
 | 
						|
      label: "",
 | 
						|
      controlsClassName: "pgadmin-controls col-sm-12 SQL",
 | 
						|
      extraClasses: [],
 | 
						|
      helpMessage: null
 | 
						|
    },
 | 
						|
    template: _.template([
 | 
						|
      '<div class="<%=controlsClassName%>">',
 | 
						|
      '  <textarea class="<%=Backform.controlClassName%> <%=extraClasses.join(\' \')%>" name="<%=name%>" placeholder="<%-placeholder%>" <%=disabled ? "disabled" : ""%> <%=required ? "required" : ""%>><%-value%></textarea>',
 | 
						|
      '  <% if (helpMessage && helpMessage.length) { %>',
 | 
						|
      '    <span class="<%=Backform.helpMessageClassName%>"><%=helpMessage%></span>',
 | 
						|
      '  <% } %>',
 | 
						|
      '</div>'
 | 
						|
    ].join("\n")),
 | 
						|
    /*
 | 
						|
     * Initialize the SQL Tab control properly
 | 
						|
     */
 | 
						|
    initialize: function(o) {
 | 
						|
      Backform.Control.prototype.initialize.apply(this, arguments);
 | 
						|
 | 
						|
      // Save the required information for using it later.
 | 
						|
      this.dialog = o.dialog;
 | 
						|
      this.tabIndex = o.tabIndex;
 | 
						|
 | 
						|
      /*
 | 
						|
       * We will listen to the tab change event to check, if the SQL tab has
 | 
						|
       * been clicked or, not.
 | 
						|
       */
 | 
						|
      this.model.on('pg-property-tab-changed', this.onTabChange, this);
 | 
						|
    },
 | 
						|
    getValueFromDOM: function() {
 | 
						|
        return this.formatter.toRaw(this.$el.find("textarea").val(), this.model);
 | 
						|
    },
 | 
						|
    render: function() {
 | 
						|
      if (this.sqlCtrl) {
 | 
						|
        delete this.sqlCtrl;
 | 
						|
        this.sqlCtrl = null;
 | 
						|
        this.$el.empty();
 | 
						|
      }
 | 
						|
      // Use the Backform Control's render function
 | 
						|
      Backform.Control.prototype.render.apply(this, arguments);
 | 
						|
 | 
						|
      this.sqlCtrl = CodeMirror.fromTextArea(
 | 
						|
        (this.$el.find("textarea")[0]), {
 | 
						|
        lineNumbers: true,
 | 
						|
        lineWrapping: true,
 | 
						|
        mode: "text/x-pgsql",
 | 
						|
        readOnly: true
 | 
						|
      });
 | 
						|
 | 
						|
      return this;
 | 
						|
    },
 | 
						|
    onTabChange: function(obj) {
 | 
						|
 | 
						|
      // Fetch the information only if the SQL tab is visible at the moment.
 | 
						|
      if (this.dialog && obj.shown == this.tabIndex) {
 | 
						|
 | 
						|
        // We will send request to sever only if something is changed in model
 | 
						|
        if(this.model.sessChanged()) {
 | 
						|
 | 
						|
          var self = this,
 | 
						|
            node = self.field.get('schema_node'),
 | 
						|
            msql_url = node.generate_url.apply(
 | 
						|
                node, [
 | 
						|
                  null, 'msql', this.field.get('node_data'), !self.model.isNew(),
 | 
						|
                  this.field.get('node_info')
 | 
						|
                ]);
 | 
						|
 | 
						|
          // Fetching the modified SQL
 | 
						|
          self.model.trigger('pgadmin-view:msql:fetching', self.method, node);
 | 
						|
 | 
						|
          $.ajax({
 | 
						|
            url: msql_url,
 | 
						|
            type: 'GET',
 | 
						|
            cache: false,
 | 
						|
            data: self.model.toJSON(true, 'GET'),
 | 
						|
            dataType: "json",
 | 
						|
            contentType: "application/json"
 | 
						|
          }).done(function(res) {
 | 
						|
            self.sqlCtrl.clearHistory();
 | 
						|
            self.sqlCtrl.setValue(res.data);
 | 
						|
          }).fail(function() {
 | 
						|
            self.model.trigger('pgadmin-view:msql:error', self.method, node, arguments);
 | 
						|
          }).always(function() {
 | 
						|
            self.model.trigger('pgadmin-view:msql:fetched', self.method, node, arguments);
 | 
						|
          });
 | 
						|
        } else {
 | 
						|
          this.sqlCtrl.clearHistory();
 | 
						|
          this.sqlCtrl.setValue(window.pgAdmin.Browser.messages.SQL_NO_CHANGE);
 | 
						|
        }
 | 
						|
        this.sqlCtrl.refresh.apply(this.sqlCtrl);
 | 
						|
      }
 | 
						|
    },
 | 
						|
    remove: function() {
 | 
						|
      if (this.sqlCtrl) {
 | 
						|
        delete this.sqlCtrl;
 | 
						|
        this.sqlCtrl = null;
 | 
						|
        this.$el.empty();
 | 
						|
      }
 | 
						|
      this.model.off('pg-property-tab-changed', this.onTabChange, this);
 | 
						|
      Backform.Control.__super__.remove.apply(this, arguments);
 | 
						|
    }
 | 
						|
});
 | 
						|
 | 
						|
  /*
 | 
						|
   * Integer input Control functionality just like backgrid
 | 
						|
   */
 | 
						|
  var IntegerControl = Backform.IntegerControl = Backform.InputControl.extend({
 | 
						|
    defaults: {
 | 
						|
      type: "number",
 | 
						|
      label: "",
 | 
						|
      min: undefined,
 | 
						|
      max: undefined,
 | 
						|
      maxlength: 255,
 | 
						|
      extraClasses: [],
 | 
						|
      helpMessage: null
 | 
						|
    },
 | 
						|
    template: _.template([
 | 
						|
      '<label class="<%=Backform.controlLabelClassName%>"><%=label%></label>',
 | 
						|
      '<div class="<%=Backform.controlsClassName%>">',
 | 
						|
      '  <input type="<%=type%>" class="<%=Backform.controlClassName%> <%=extraClasses.join(\' \')%>" name="<%=name%>" min="<%=min%>" max="<%=max%>"maxlength="<%=maxlength%>" value="<%-value%>" placeholder="<%-placeholder%>" <%=disabled ? "disabled" : ""%> <%=required ? "required" : ""%> />',
 | 
						|
      '  <% if (helpMessage && helpMessage.length) { %>',
 | 
						|
      '    <span class="<%=Backform.helpMessageClassName%>"><%=helpMessage%></span>',
 | 
						|
      '  <% } %>',
 | 
						|
      '</div>'
 | 
						|
    ].join("\n")),
 | 
						|
    events: {
 | 
						|
      "change input": "checkInt",
 | 
						|
      "focus input": "clearInvalid"
 | 
						|
    },
 | 
						|
    checkInt: function(e) {
 | 
						|
      var field = _.defaults(this.field.toJSON(), this.defaults),
 | 
						|
          attrArr = this.field.get("name").split('.'),
 | 
						|
          name = attrArr.shift(),
 | 
						|
          value = this.getValueFromDOM(),
 | 
						|
          min_value = field.min,
 | 
						|
          max_value = field.max,
 | 
						|
          isValid = true,
 | 
						|
          intPattern = new RegExp("^-?[0-9]*$"),
 | 
						|
          isMatched = intPattern.test(value);
 | 
						|
 | 
						|
      // Below logic will validate input
 | 
						|
      if (!isMatched) {
 | 
						|
        isValid = false;
 | 
						|
        this.model.errorModel.unset(name);
 | 
						|
        this.model.errorModel.set(
 | 
						|
            name,
 | 
						|
            S(pgAdmin.Browser.messages.MUST_BE_INT).sprintf(
 | 
						|
              field.label
 | 
						|
              ).value()
 | 
						|
            );
 | 
						|
      }
 | 
						|
 | 
						|
      // Below will check if entered value is in-between min & max range
 | 
						|
      if (isValid && (!_.isUndefined(min_value) && value < min_value)) {
 | 
						|
        isValid = false;
 | 
						|
        this.model.errorModel.unset(name);
 | 
						|
        this.model.errorModel.set(
 | 
						|
            name,
 | 
						|
            S(pgAdmin.Browser.messages.MUST_GR_EQ).sprintf(
 | 
						|
              field.label,
 | 
						|
              min_value
 | 
						|
              ).value()
 | 
						|
            );
 | 
						|
      }
 | 
						|
 | 
						|
      if (isValid && (!_.isUndefined(max_value) && value > max_value)) {
 | 
						|
        isValid = false;
 | 
						|
        this.model.errorModel.unset(name);
 | 
						|
        this.model.errorModel.set(
 | 
						|
            name,
 | 
						|
            S(pgAdmin.Browser.messages.MUST_LESS_EQ).sprintf(
 | 
						|
              field.label,
 | 
						|
              max_value
 | 
						|
              ).value()
 | 
						|
            );
 | 
						|
      }
 | 
						|
 | 
						|
      // After validation we need to set that value into model (only if all flags are true)
 | 
						|
      if (isValid) {
 | 
						|
        this.stopListening(this.model, "change:" + name, this.render);
 | 
						|
        this.model.set(name, value);
 | 
						|
        this.listenTo(this.model, "change:" + name, this.render);
 | 
						|
      } else {
 | 
						|
        if (this.model.collection || this.model.handler) {
 | 
						|
          (this.model.collection || this.model.handler).trigger(
 | 
						|
             'pgadmin-session:model:invalid', this.model.errorModel.get(name), this.model
 | 
						|
            );
 | 
						|
        } else {
 | 
						|
          (this.model).trigger(
 | 
						|
             'pgadmin-session:invalid', this.model.errorModel.get(name), this.model
 | 
						|
            );
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
   /*
 | 
						|
   * Numeric input Control functionality just like backgrid
 | 
						|
   */
 | 
						|
  var NumericControl = Backform.NumericControl = Backform.InputControl.extend({
 | 
						|
    defaults: {
 | 
						|
      type: "number",
 | 
						|
      label: "",
 | 
						|
      min: undefined,
 | 
						|
      max: undefined,
 | 
						|
      maxlength: 255,
 | 
						|
      extraClasses: [],
 | 
						|
      helpMessage: null
 | 
						|
    },
 | 
						|
    template: _.template([
 | 
						|
      '<label class="<%=Backform.controlLabelClassName%>"><%=label%></label>',
 | 
						|
      '<div class="<%=Backform.controlsClassName%>">',
 | 
						|
      '  <input type="<%=type%>" class="<%=Backform.controlClassName%> <%=extraClasses.join(\' \')%>" name="<%=name%>" min="<%=min%>" max="<%=max%>"maxlength="<%=maxlength%>" value="<%-value%>" placeholder="<%-placeholder%>" <%=disabled ? "disabled" : ""%> <%=required ? "required" : ""%> />',
 | 
						|
      '  <% if (helpMessage && helpMessage.length) { %>',
 | 
						|
      '    <span class="<%=Backform.helpMessageClassName%>"><%=helpMessage%></span>',
 | 
						|
      '  <% } %>',
 | 
						|
      '</div>'
 | 
						|
    ].join("\n")),
 | 
						|
    events: {
 | 
						|
      "change input": "checkNumeric",
 | 
						|
      "focus input": "clearInvalid"
 | 
						|
    },
 | 
						|
    checkNumeric: function(e) {
 | 
						|
      var field = _.defaults(this.field.toJSON(), this.defaults),
 | 
						|
          attrArr = this.field.get("name").split('.'),
 | 
						|
          name = attrArr.shift(),
 | 
						|
          value = this.getValueFromDOM(),
 | 
						|
          min_value = field.min,
 | 
						|
          max_value = field.max,
 | 
						|
          isValid = true,
 | 
						|
          intPattern = new RegExp("^-?[0-9]+(\.?[0-9]*)?$"),
 | 
						|
          isMatched = intPattern.test(value);
 | 
						|
 | 
						|
      // Below logic will validate input
 | 
						|
      if (!isMatched) {
 | 
						|
        isValid = false;
 | 
						|
        this.model.errorModel.unset(name);
 | 
						|
        this.model.errorModel.set(
 | 
						|
            name,
 | 
						|
            S(pgAdmin.Browser.messages.MUST_BE_NUM).sprintf(
 | 
						|
              field.label
 | 
						|
              ).value()
 | 
						|
            );
 | 
						|
      }
 | 
						|
 | 
						|
      // Below will check if entered value is in-between min & max range
 | 
						|
      if (isValid && (!_.isUndefined(min_value) && value < min_value)) {
 | 
						|
        isValid = false;
 | 
						|
        this.model.errorModel.unset(name);
 | 
						|
        this.model.errorModel.set(
 | 
						|
            name,
 | 
						|
            S(pgAdmin.Browser.messages.MUST_GR_EQ).sprintf(
 | 
						|
              field.label,
 | 
						|
              min_value
 | 
						|
              ).value()
 | 
						|
            );
 | 
						|
      }
 | 
						|
 | 
						|
      if (isValid && (!_.isUndefined(max_value) && value > max_value)) {
 | 
						|
        isValid = false;
 | 
						|
        this.model.errorModel.unset(name);
 | 
						|
        this.model.errorModel.set(
 | 
						|
            name,
 | 
						|
            S(pgAdmin.Browser.messages.MUST_LESS_EQ).sprintf(
 | 
						|
              field.label,
 | 
						|
              max_value
 | 
						|
              ).value()
 | 
						|
            );
 | 
						|
      }
 | 
						|
 | 
						|
      // After validation we need to set that value into model (only if all flags are true)
 | 
						|
      if (isValid) {
 | 
						|
        this.stopListening(this.model, "change:" + name, this.render);
 | 
						|
        this.model.set(name, value);
 | 
						|
        this.listenTo(this.model, "change:" + name, this.render);
 | 
						|
      } else {
 | 
						|
        if (this.model.collection || this.model.handler) {
 | 
						|
          (this.model.collection || this.model.handler).trigger(
 | 
						|
             'pgadmin-session:model:invalid', this.model.errorModel.get(name), this.model
 | 
						|
            );
 | 
						|
        } else {
 | 
						|
          (this.model).trigger(
 | 
						|
             'pgadmin-session:invalid', this.model.errorModel.get(name), this.model
 | 
						|
            );
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  ///////
 | 
						|
  // Generate a schema (as group members) based on the model's schema
 | 
						|
  //
 | 
						|
  // It will be used by the grid, properties, and dialog view generation
 | 
						|
  // functions.
 | 
						|
  var generateViewSchema = Backform.generateViewSchema = function(
 | 
						|
      node_info, Model, mode, node, treeData, noSQL, subschema
 | 
						|
      ) {
 | 
						|
    var proto = (Model && Model.prototype) || Model,
 | 
						|
        schema = subschema || (proto && proto.schema),
 | 
						|
        pgBrowser = window.pgAdmin.Browser, fields = [],
 | 
						|
        groupInfo = {};
 | 
						|
 | 
						|
    // 'schema' has the information about how to generate the form.
 | 
						|
    if (schema && _.isArray(schema)) {
 | 
						|
      var evalASFunc = evalASFunc = function(prop) {
 | 
						|
        return ((prop && proto[prop] &&
 | 
						|
              typeof proto[prop] == "function") ? proto[prop] : prop);
 | 
						|
      };
 | 
						|
      var groups = {},
 | 
						|
          server_info = node_info && ('server' in node_info) &&
 | 
						|
            pgBrowser.serverInfo && pgBrowser.serverInfo[node_info.server._id];
 | 
						|
 | 
						|
      _.each(schema, function(s) {
 | 
						|
        // Do we understand - what control, we're creating
 | 
						|
        // here?
 | 
						|
        if (s.type == 'group') {
 | 
						|
          var ver_in_limit = (_.isUndefined(server_info) ? true :
 | 
						|
              ((_.isUndefined(s.server_type) ? true :
 | 
						|
                (server_info.type in s.server_type)) &&
 | 
						|
              (_.isUndefined(s.min_version) ? true :
 | 
						|
               (server_info.version >= s.min_version)) &&
 | 
						|
              (_.isUndefined(s.max_version) ? true :
 | 
						|
               (server_info.version <= s.max_version))));
 | 
						|
          groupInfo[s.id] = {
 | 
						|
            label: s.label || s.id,
 | 
						|
            version_compatible: ver_in_limit,
 | 
						|
            visible: !s.mode || (
 | 
						|
              s && s.mode && _.isObject(s.mode) &&
 | 
						|
                _.indexOf(s.mode, mode) != -1) && evalASFunc(s.visible) || true
 | 
						|
          };
 | 
						|
          return;
 | 
						|
        }
 | 
						|
 | 
						|
        if (!s.mode || (s && s.mode && _.isObject(s.mode) &&
 | 
						|
          _.indexOf(s.mode, mode) != -1)) {
 | 
						|
          // Each field is kept in specified group, or in
 | 
						|
          // 'General' category.
 | 
						|
          var group = s.group || pgBrowser.messages.GENERAL_CATEGORY,
 | 
						|
              control = s.control || Backform.getMappedControl(s.type, mode),
 | 
						|
              cell =  s.cell || Backform.getMappedControl(s.type, 'cell');
 | 
						|
 | 
						|
          if (control == null) {
 | 
						|
            return;
 | 
						|
          }
 | 
						|
 | 
						|
          // Generate the empty group list (if not exists)
 | 
						|
          groups[group] = (groups[group] || []);
 | 
						|
          var ver_in_limit = (_.isUndefined(server_info) ? true :
 | 
						|
                ((_.isUndefined(s.server_type) ? true :
 | 
						|
                  (server_info.type in s.server_type)) &&
 | 
						|
                (_.isUndefined(s.min_version) ? true :
 | 
						|
                 (server_info.version >= s.min_version)) &&
 | 
						|
                (_.isUndefined(s.max_version) ? true :
 | 
						|
                 (server_info.version <= s.max_version)))),
 | 
						|
              disabled = ((mode == 'properties') || !ver_in_limit),
 | 
						|
              schema_node = (s.node && _.isString(s.node) &&
 | 
						|
                  s.node in pgBrowser.Nodes &&  pgBrowser.Nodes[s.node]) || node;
 | 
						|
 | 
						|
          var o = _.extend(_.clone(s), {
 | 
						|
            name: s.id,
 | 
						|
            // This can be disabled in some cases (if not hidden)
 | 
						|
 | 
						|
            disabled: (disabled ? true : evalASFunc(s.disabled)),
 | 
						|
            editable: _.isUndefined(s.editable) ?
 | 
						|
                pgAdmin.editableCell : evalASFunc(s.editable),
 | 
						|
            subnode: ((_.isString(s.model) && s.model in pgBrowser.Nodes) ?
 | 
						|
                pgBrowser.Nodes[s.model].model : s.model),
 | 
						|
            canAdd: (disabled ? false : evalASFunc(s.canAdd)),
 | 
						|
            canAddRow: (disabled ? false : evalASFunc(s.canAddRow)),
 | 
						|
            canEdit: (disabled ? false : evalASFunc(s.canEdit)),
 | 
						|
            canDelete: (disabled ? false : evalASFunc(s.canDelete)),
 | 
						|
            canEditRow: (disabled ? false : evalASFunc(s.canEditRow)),
 | 
						|
            canDeleteRow: (disabled ? false : evalASFunc(s.canDeleteRow)),
 | 
						|
            transform: evalASFunc(s.transform),
 | 
						|
            mode: mode,
 | 
						|
            control: control,
 | 
						|
            cell: cell,
 | 
						|
            node_info: node_info,
 | 
						|
            schema_node: schema_node,
 | 
						|
            // Do we need to show this control in this mode?
 | 
						|
            visible: evalASFunc(s.visible),
 | 
						|
            node: node,
 | 
						|
            node_data: treeData,
 | 
						|
            version_compatible: ver_in_limit
 | 
						|
          });
 | 
						|
          delete o.id;
 | 
						|
 | 
						|
          // Temporarily store in dictionary format for
 | 
						|
          // utilizing it later.
 | 
						|
          groups[group].push(o);
 | 
						|
 | 
						|
          if (s.type == 'nested') {
 | 
						|
            delete o.name;
 | 
						|
            delete o.cell;
 | 
						|
 | 
						|
            o.schema = Backform.generateViewSchema(
 | 
						|
                node_info, Model, mode, node, treeData, true, s.schema
 | 
						|
                );
 | 
						|
            o.control = o.control || 'tab';
 | 
						|
          }
 | 
						|
        }
 | 
						|
      });
 | 
						|
 | 
						|
      // Do we have fields to genreate controls, which we
 | 
						|
      // understand?
 | 
						|
      if (_.isEmpty(groups)) {
 | 
						|
        return null;
 | 
						|
      }
 | 
						|
 | 
						|
      if (!noSQL && node && node.hasSQL && (mode == 'create' || mode == 'edit')) {
 | 
						|
        groups[pgBrowser.messages.SQL_TAB] = [{
 | 
						|
            name: 'sql',
 | 
						|
            visible: true,
 | 
						|
            disabled: false,
 | 
						|
            type: 'text',
 | 
						|
            control: 'sql-tab',
 | 
						|
            node_info: node_info,
 | 
						|
            schema_node: node,
 | 
						|
            node_data: treeData
 | 
						|
        }];
 | 
						|
      }
 | 
						|
 | 
						|
      // Create an array from the dictionary with proper required
 | 
						|
      // structure.
 | 
						|
      _.each(groups, function(val, key) {
 | 
						|
        fields.push(_.extend({
 | 
						|
          label: key, fields: val
 | 
						|
        }, (groupInfo[key] || {version_compatible: true, visible: true})));
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    return fields;
 | 
						|
  };
 | 
						|
 | 
						|
  /*
 | 
						|
   *  Backform Select2 control.
 | 
						|
   */
 | 
						|
  var Select2Control = Backform.Select2Control = Backform.SelectControl.extend({
 | 
						|
    render: function() {
 | 
						|
      Backform.SelectControl.prototype.render.apply(this, arguments);
 | 
						|
 | 
						|
      var opts = this.field.toJSON();
 | 
						|
      var select2Opts = _.defaults(
 | 
						|
        {}, opts.select2, (this.defaults && this.defaults.select2) || {}
 | 
						|
      );
 | 
						|
 | 
						|
      /*
 | 
						|
       * Add empty option as Select2 requires any empty '<option><option>' for
 | 
						|
       * some of its functionality to work and initialize select2 control.
 | 
						|
       */
 | 
						|
      var $select = this.$el.find("select");
 | 
						|
      $select.prepend($('<option></option>'));
 | 
						|
      $select.select2(select2Opts);
 | 
						|
 | 
						|
      return this;
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  var FieldsetControl = Backform.FieldsetControl = Backform.Fieldset.extend({
 | 
						|
    initialize: function(opts) {
 | 
						|
      Backform.Control.prototype.initialize.apply(
 | 
						|
        this, arguments
 | 
						|
        );
 | 
						|
      Backform.Dialog.prototype.initialize.apply(
 | 
						|
        this, [{schema: opts.field.get('schema')}]
 | 
						|
        );
 | 
						|
      this.dialog = opts.dialog;
 | 
						|
      this.tabIndex = opts.tabIndex;
 | 
						|
 | 
						|
      // Listen to the dependent fields in the model for any change
 | 
						|
      var deps = this.field.get('deps');
 | 
						|
      var self = this;
 | 
						|
 | 
						|
      if (deps && _.isArray(deps)) {
 | 
						|
        _.each(deps, function(d) {
 | 
						|
          attrArr = d.split('.');
 | 
						|
          name = attrArr.shift();
 | 
						|
          self.listenTo(self.model, "change:" + name, self.render);
 | 
						|
        });
 | 
						|
      }
 | 
						|
    },
 | 
						|
    // Render using Backform.Fieldset (only if this control is visible)
 | 
						|
    orig_render: Backform.Fieldset.prototype.render,
 | 
						|
    render: function() {
 | 
						|
      var field = _.defaults(this.field.toJSON(), this.defaults),
 | 
						|
          evalF = function(f, d, m) {
 | 
						|
            return (_.isFunction(f) ? !!f.apply(d, [m]) : !!f);
 | 
						|
          };
 | 
						|
 | 
						|
      if (!field.version_compatible ||
 | 
						|
          !evalF(field.visible, field, this.model)) {
 | 
						|
        this.cleanup();
 | 
						|
        this.$el.empty()
 | 
						|
      } else {
 | 
						|
        this.orig_render.apply(this, arguments);
 | 
						|
      }
 | 
						|
      return this;
 | 
						|
    },
 | 
						|
    formatter: function() {},
 | 
						|
    cleanup: function() {
 | 
						|
      Backform.Fieldset.prototype.cleanup.apply(this);
 | 
						|
    },
 | 
						|
    remove: function() {
 | 
						|
      Backform.Control.prototype.remove.apply(this, arguments);
 | 
						|
      Backform.Dialog.prototype.remove.apply(this, arguments);
 | 
						|
    },
 | 
						|
    className: function() {
 | 
						|
      return 'set-group';
 | 
						|
    },
 | 
						|
    tabPanelClassName: function() {
 | 
						|
      return Backform.tabClassName;
 | 
						|
    },
 | 
						|
    fieldsetClass: 'inline-fieldset',
 | 
						|
    legendClass: '',
 | 
						|
    contentClass: '',
 | 
						|
    collapse: false
 | 
						|
  });
 | 
						|
 | 
						|
 | 
						|
  // Backform Tab Control (in bootstrap tabbular)
 | 
						|
  // A collection of field models.
 | 
						|
  var TabControl = Backform.TabControl = Backform.FieldsetControl.extend({
 | 
						|
    tagName: "div",
 | 
						|
    className: 'inline-tab-panel',
 | 
						|
    tabPanelClassName: 'inline-tab-panel',
 | 
						|
    initialize: function(opts) {
 | 
						|
      Backform.FieldsetControl.prototype.initialize.apply(
 | 
						|
        this, arguments
 | 
						|
        );
 | 
						|
      this.tabIndex = (opts.tabIndex || parseInt(Math.random() * 1000)) + 1;
 | 
						|
    },
 | 
						|
    // Render using Backform.Dialog (tabular UI) (only if this control is
 | 
						|
    // visible).
 | 
						|
    orig_render: Backform.Dialog.prototype.render,
 | 
						|
    template: Backform.Dialog.prototype.template
 | 
						|
  });
 | 
						|
 | 
						|
 | 
						|
  // Backform Tab Control (in bootstrap tabbular)
 | 
						|
  // A collection of field models.
 | 
						|
  var PlainFieldsetControl = Backform.PlainFieldsetControl = Backform.FieldsetControl.extend({
 | 
						|
    initialize: function(opts) {
 | 
						|
      Backform.FieldsetControl.prototype.initialize.apply(
 | 
						|
        this, arguments
 | 
						|
        );
 | 
						|
    },
 | 
						|
    template: {
 | 
						|
      'header': _.template([
 | 
						|
        '<fieldset class="<%=fieldsetClass%>" <%=disabled ? "disabled" : ""%>>',
 | 
						|
        ' <% if (legend != false) { %>',
 | 
						|
        '  <legend class="<%=legendClass%>" <%=collapse ? "data-toggle=\'collapse\'" : ""%> data-target="#<%=cId%>"><%=collapse ? "<span class=\'caret\'></span>" : "" %></legend>',
 | 
						|
        ' <% } %>',
 | 
						|
        '</fieldset>'
 | 
						|
      ].join("\n")),
 | 
						|
      'content': _.template(
 | 
						|
        '  <div id="<%= cId %>" class="<%=contentClass%>"></div>'
 | 
						|
    )},
 | 
						|
    fieldsetClass: 'inline-fieldset-without-border',
 | 
						|
    legend: false,
 | 
						|
  });
 | 
						|
 | 
						|
  /*
 | 
						|
   * Control For Code Mirror SQL text area.
 | 
						|
   */
 | 
						|
  var SqlFieldControl = Backform.SqlFieldControl = Backform.TextareaControl.extend({
 | 
						|
 | 
						|
    defaults: {
 | 
						|
      label: "",
 | 
						|
      extraClasses: ['sql_field_height_300'], // Add default control height
 | 
						|
      helpMessage: null,
 | 
						|
      maxlength: 4096,
 | 
						|
      rows: undefined
 | 
						|
    },
 | 
						|
 | 
						|
    // Customize template to add new styles
 | 
						|
    template: _.template([
 | 
						|
      '<label class="<%=Backform.controlLabelClassName%>"><%=label%></label>',
 | 
						|
      '<div class="<%=Backform.controlsClassName%> sql_field_layout <%=extraClasses.join(\' \')%>">',
 | 
						|
      '  <textarea ',
 | 
						|
      '    class="<%=Backform.controlClassName%> " name="<%=name%>"',
 | 
						|
      '    maxlength="<%=maxlength%>" placeholder="<%-placeholder%>" <%=disabled ? "disabled" : ""%>',
 | 
						|
      '    rows=<%=rows%>',
 | 
						|
      '    <%=required ? "required" : ""%>><%-value%></textarea>',
 | 
						|
      '  <% if (helpMessage && helpMessage.length) { %>',
 | 
						|
      '    <span class="<%=Backform.helpMessageClassName%>"><%=helpMessage%></span>',
 | 
						|
      '  <% } %>',
 | 
						|
      '</div>'
 | 
						|
    ].join("\n")),
 | 
						|
 | 
						|
    /*
 | 
						|
     * Initialize the SQL Field control properly
 | 
						|
     */
 | 
						|
    initialize: function(o) {
 | 
						|
      Backform.TextareaControl.prototype.initialize.apply(this, arguments);
 | 
						|
      this.sqlCtrl = null;
 | 
						|
 | 
						|
      // There is an issue with the Code Mirror SQL.
 | 
						|
      //
 | 
						|
      // It does not initialize the code mirror object completely when the
 | 
						|
      // referenced textarea is hidden (not visible), hence - we need to
 | 
						|
      // refresh the code mirror object on 'pg-property-tab-changed' event to
 | 
						|
      // make it work properly.
 | 
						|
      this.listenTo(this.model, 'pg-property-tab-changed', this.refreshTextArea);
 | 
						|
    },
 | 
						|
 | 
						|
    getValueFromDOM: function() {
 | 
						|
      return this.sqlCtrl.getValue();
 | 
						|
    },
 | 
						|
 | 
						|
    render: function() {
 | 
						|
      // Clean up the existing sql control
 | 
						|
      if (this.sqlCtrl) {
 | 
						|
        delete this.sqlCtrl;
 | 
						|
        this.sqlCtrl = null;
 | 
						|
        this.$el.empty();
 | 
						|
      }
 | 
						|
 | 
						|
      // Use the Backform TextareaControl's render function
 | 
						|
      Backform.TextareaControl.prototype.render.apply(this, arguments);
 | 
						|
 | 
						|
      var field = _.defaults(this.field.toJSON(), this.defaults),
 | 
						|
          attributes = this.model.toJSON(),
 | 
						|
          attrArr = field.name.split('.'),
 | 
						|
          name = attrArr.shift(),
 | 
						|
          path = attrArr.join('.'),
 | 
						|
          rawValue = this.keyPathAccessor(attributes[name], path),
 | 
						|
          data = _.extend(field, {
 | 
						|
            rawValue: rawValue,
 | 
						|
            value: this.formatter.fromRaw(rawValue, this.model),
 | 
						|
            attributes: attributes,
 | 
						|
            formatter: this.formatter
 | 
						|
          }),
 | 
						|
          evalF = function(f, d, m) {
 | 
						|
            return (_.isFunction(f) ? !!f.apply(d, [m]) : !!f);
 | 
						|
          };
 | 
						|
 | 
						|
      // Evaluate the disabled, visible option
 | 
						|
      var isDisabled = evalF(data.disabled, data, this.model),
 | 
						|
          isVisible = evalF(data.visible, data, this.model),
 | 
						|
          self = this;
 | 
						|
 | 
						|
      self.sqlCtrl = CodeMirror.fromTextArea(
 | 
						|
            (self.$el.find("textarea")[0]), {
 | 
						|
            lineNumbers: true,
 | 
						|
            mode: "text/x-sql",
 | 
						|
            readOnly: isDisabled
 | 
						|
          });
 | 
						|
 | 
						|
      if (!isVisible)
 | 
						|
        self.$el.addClass(Backform.hiddenClassname);
 | 
						|
 | 
						|
      var self = this;
 | 
						|
      // Refresh SQL Field to refresh the control lazily after it renders
 | 
						|
      setTimeout(function() {
 | 
						|
        self.refreshTextArea.apply(self);
 | 
						|
      }, 0);
 | 
						|
 | 
						|
      return self;
 | 
						|
    },
 | 
						|
 | 
						|
    refreshTextArea: function() {
 | 
						|
      if (this.sqlCtrl) {
 | 
						|
        this.sqlCtrl.refresh();
 | 
						|
      }
 | 
						|
    },
 | 
						|
 | 
						|
    remove: function() {
 | 
						|
      // Clean up the sql control
 | 
						|
      if (this.sqlCtrl) {
 | 
						|
        delete this.sqlCtrl;
 | 
						|
        this.sqlCtrl = null;
 | 
						|
        this.$el.empty();
 | 
						|
      }
 | 
						|
 | 
						|
      this.stopListening(this.model, "pg-property-tab-changed", this.refreshTextArea);
 | 
						|
 | 
						|
      Backform.TextareaControl.prototype.remove.apply(this, arguments);
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  // We will use this control just as a annotate in Backform
 | 
						|
  var NoteControl = Backform.NoteControl = Backform.Control.extend({
 | 
						|
    defaults: {
 | 
						|
      label: window.pgAdmin.Browser.messages.NOTE_CTRL_LABEL,
 | 
						|
      text: '',
 | 
						|
      extraClasses: [],
 | 
						|
      noteClass: 'backform_control_notes'
 | 
						|
    },
 | 
						|
    template: _.template([
 | 
						|
      '<div class="<%=noteClass%> col-xs-12 <%=extraClasses.join(\' \')%>">',
 | 
						|
      '<label class="control-label"><%=label%>:</label>',
 | 
						|
      '<span><%=text%></span></div>'
 | 
						|
    ].join("\n"))
 | 
						|
  });
 | 
						|
 | 
						|
  return Backform;
 | 
						|
}));
 |