Print the App Schema (Detailed)

Using this code in your BODY, STYLES, and SCRIPT tabs to allow you to print your Fulcrum app and share it with others as a PDF. You do not need to make it active. Keep it inactive and download or print from the advanced report builder page.

<div class="header">
  <img id="form-image" src="https://learn.fulcrumapp.com/img/branding/fulcrum-icon.png" />
  <div>
    <h2 id="form-name"></h2>
    <div id="form-description"></div>
  </div>
</div>
<div id="elements"></div>

<script>
const form = <%- TOJSON(API(`/forms/${QUERYVALUE(`SELECT form_id FROM forms WHERE name = '${form.name}'`)}`).form) %>;
const choiceLists = <%- TOJSON(API(`/choice_lists`).choice_lists) %>;

document.getElementById('form-name').innerHTML = form.name;
document.getElementById('form-description').innerHTML = form.description;
document.getElementById('form-image').src = form.image ? form.image : "https://learn.fulcrumapp.com/img/branding/fulcrum-icon.png";

const images = {
  'AddressField': '',
  'AudioField': '',
  'BarcodeField': '',
  'CalculatedField': '',
  'ChoiceField': '',
  'ClassificationField': '',
  'DateTimeField': '',
  'HyperlinkField': '',
  'Label': '',
  'MultiChoice': '',
  'Numeric': '',
  'PhotoField': '',
  'RecordLinkField': '',
  'Repeatable': '',
  'Section': '',
  'SignatureField': '',
  'TextField': '',
  'TimeField': '',
  'YesNoField': '',
  'VideoField': ''
};

document.getElementById('elements').appendChild(parseElements(form.elements))

function formatDataName(input){
  if (input.length > 25) {
    return input.substring(0, 25) + '...';
  }
  return input;
}

function parseElements(elements) { // takes an elements array and turns it into a <ol>
  let ul = document.createElement('ul');
  for (let i=0; i<elements.length; i++) {
    ul.appendChild(parseElement(elements[i]));
  }
  return ul;
}

function getImage(element) {
  if (element.type == 'TextField' && element.numeric == true) {
    return images['Numeric'];
  } else if (element.type == 'ChoiceField' && element.multiple) {
    return images['MultiChoice'];
  } else {
    return images[element.type];
  }
}

function getType(element) {
  if (element.type == 'TextField' && element.numeric == true) {
    return `NumericField (${element.format})`;
  } else if (element.type == 'ChoiceField') {
    let type = element.multiple ? 'MultiChoiceField' : 'ChoiceField';
    if (element.choices) {
      type += ' (inline)'
    } else if (element.choice_list_id) {
      type += ' (pre-defined)'
    }
    return type;
  } else {
    return element.type;
  }
}

function getChoiceTable(element) {
  if (element.choices && element.choices.length > 0) {
    let table = `<h5>Labels & Values</h5><table>
      <tr>
        <th>Label</th>
        <th>Value</th>
      </tr>
    `;
    element.choices.forEach(choice => {
      table += `<tr>
        <td>${choice.label}</td>
        <td>${choice.value}</td>
      </tr>`;
    });
    table += '</table>';
    return table;
  }
}

function getRemoteChoiceTable(id) {
  let table = `<h5>Labels & Values</h5><table
    <tr>
      <th>Label</th>
      <th>Value</th>
    </tr>
  `;
  choiceLists.forEach(list => {
    if (list.id == id) {
      list.choices.forEach(choice => {
        table += `<tr>
          <td>${choice.label}</td>
          <td>${choice.value}</td>
        </tr>`;
      });
    }
  });
  table += '</table>';
  return table;
}

function getYesNoTable(element) {
  let table = `<h5>Labels & Values</h5><table
    <tr>
      <th>Label</th>
      <th>Value</th>
    </tr>
    <tr>
      <td>${element.positive.label}</td>
      <td>${element.positive.value}</td>
    </tr>
    <tr>
      <td>${element.negative.label}</td>
      <td>${element.negative.value}</td>
    </tr>
  `;
  if (element.neutral_enabled) {
    table += ` <tr>
      <td>${element.neutral.label}</td>
      <td>${element.neutral.value}</td>
    </tr>`;
  }
  table += '</table>';
  return table;
}

function getVizConditionsTable(element) {
  const conditions = element.visible_conditions;
  let table = `<h5>Visibility Conditions (${element.visible_conditions_type}, ${element.visible_conditions_behavior} data)</h5><table>
    <tr>
      <th>Field</th>
      <th>Operator</th>
      <th>Value</th>
    </tr>
  `;
  conditions.forEach(condition => {
    table += `<tr>
      <td>${getObjects(form, 'key', condition.field_key)[0].label}</td>
      <td>${condition.operator}</td>
      <td>${condition.value}</td>
    </tr>`;
  });
  table += '</table>';
  return table;
}

function getRequiredConditionsTable(element) {
  const conditions = element.required_conditions;
  let table = `<h5>Required Conditions (${element.required_conditions_type})</h5><table>
    <tr>
      <th>Field</th>
      <th>Operator</th>
      <th>Value</th>
    </tr>
  `;
  conditions.forEach(condition => {
    table += `<tr>
      <td>${getObjects(form, 'key', condition.field_key)[0].label}</td>
      <td>${condition.operator}</td>
      <td>${condition.value}</td>
    </tr>`;
  });
  table += '</table>';
  return table;
}

function getObjects(obj, key, val) {
  var objects = [];
  for (var i in obj) {
    if (!obj.hasOwnProperty(i)) continue;
    if (typeof obj[i] == 'object') {
      objects = objects.concat(getObjects(obj[i], key, val));
    } else
      //if key matches and value matches or if key matches and value is not passed (eliminating the case where key matches but passed value does not)
      if (i == key && obj[i] == val || i == key && val == '') { //
        objects.push(obj);
      } else if (obj[i] == val && key == '') {
      //only add if the object is not already in the array
      if (objects.lastIndexOf(obj) == -1) {
        objects.push(obj);
      }
    }
  }
  return objects;
}

function parseElement(element) { // takes an element object and turns it into a <li>
  let li = document.createElement('li');

  if (element.type == 'Section' || element.type == 'Repeatable') {
    li.innerHTML = `
      <span class='icon' style='background-image: url(${getImage(element)})'></span>
      <h3>${element.label}</h3>
      <div class='field-description'>${element.description ? element.description : ''}</div>
      <table>
        <tr>
          ${element.type == 'Repeatable' ? '<th>Data Name</th>' : ''}
          ${element.display ? '<th>Display</th>' : ''}
          ${element.type == 'Repeatable' ? '<th>Required</th>' : ''}
          ${element.type == 'Repeatable' ? '<th>Location Required</th>' : ''}
          ${element.required_conditions ? '<th>Req Rules</th>' : ''}
          <th>Hidden</th>
          ${element.min_length ? '<th>Min Count</th>' : ''}
          ${element.max_length ? '<th>Max Count</th>' : ''}
        </tr>
        <tr>
          ${element.type == 'Repeatable' ? '<td>' + element.data_name + '</td>' : ''}
          ${element.display ? '<td>' + element.display + '</td>' : ''}
          ${element.type == 'Repeatable' ? '<td>' + element.required + '</td>' : ''}
          ${element.type == 'Repeatable' ? '<td>' + element.geometry_required + '</td>' : ''}
          ${element.required_conditions ? '<td>' + element.required_conditions + '</td>' : ''}
          <td>${element.hidden}</td>
          ${element.min_length ? '<td>' + element.min_length + '</td>' : ''}
          ${element.max_length ? '<td>' + element.max_length + '</td>' : ''}
        </tr>
      </table>
    `;
  } else {
    li.innerHTML = `
      <span class='icon' style='background-image: url(${getImage(element)})'></span>
      <h4>${element.label}</h4>
      <div class='field-description'>${element.description ? element.description : ''}</div>
      <table>
        <tr>
          <th>Data Name</th>
          <th>Required</th>
          <th>Type</th>
          <th>Disabled</th>
          <th>Hidden</th>
          ${element.default_value ? '<th>Default</th>' : ''}
          ${element.default_previous_value ? '<th>Previous Value</th>' : ''}
          ${element.min_length ? '<th>Min Length</th>' : ''}
          ${element.max_length ? '<th>Max Length</th>' : ''}
        </tr>
        <tr>
          <td>${formatDataName(element.data_name)}</td>
          <td>${element.required}</td>
          <td>${getType(element)}</td>
          <td>${element.disabled}</td>
          <td>${element.hidden}</td>
          ${element.default_value ? '<td>' + element.default_value + '</td>' : ''}
          ${element.default_previous_value ? '<td>' + element.default_previous_value + '</td>' : ''}
          ${element.min_length ? '<td>' + element.min_length + '</td>' : ''}
          ${element.max_length ? '<td>' + element.max_length + '</td>' : ''}
        </tr>
      </table>
      ${element.visible_conditions ? getVizConditionsTable(element) : ''}
      ${(element.required_conditions && element.required_conditions.length > 0) ? getRequiredConditionsTable(element) : ''}
      ${element.choices ? getChoiceTable(element) : ''}
      ${element.choice_list_id ? getRemoteChoiceTable(element.choice_list_id) : ''}
      ${element.type == 'YesNoField' ? getYesNoTable(element) : ''}
    `;
  }

  if (element.elements) {
    li.appendChild(parseElements(element.elements));
  }
  return li;
}
</script>
Leave Blank
Leave Blank
h1, h2, h3, h4, h5, h6 {
  margin: 0;
  padding: 0;
}

h5 {
  margin: 5px 0px;
}

.header {
  display: flex;
  align-items: center;
}

#form-image {
  height: 50px;
  padding-right: 10px;
}

ul {
  padding-inline-start: 20px;
  list-style-type: none;
}

li {
  border: 1px solid #d8d8d8;
  margin: 6px 0px;
  padding: 6px;
  background: #f7f7f7;
  /* page-break-inside: avoid; */
}

#elements {
  margin-top: 20px;
}

#elements ul:first-child {
  padding-inline-start: 0px;
}

#form-description {
  padding-top: 2px;
}

.field-description {
  font-size: 11px;
  font-style: italic;
  margin-top: 5px;
}

.bold {
  font-weight: bold;
}

.icon {
  width: 16px;
  height: 16px;
  display: block;
  float: left;
  background-repeat: no-repeat;
  background-size: contain;
  margin-right: 5px;
  opacity: 0.5;
}

table {
  margin-top: 5px;
  width: 100%;
}

table, th, td {
  border: 1px solid #d8d8d8;
  border-collapse: collapse;
  font-size: 10px;
  text-align: left;
}

th, td {
  padding: 4px;
}
DATA.config.advanced = true;

const REPORT_CONFIG = {
  'hidden_fields': [],
  'field.empty_value': '',
  'header.enabled': true,
  'header.form.name': true,
  'header.record.id': true,
  'footer.enabled': true,
  'footer.timestamp': true,
  'footer.organization_image': true,
  'footer.organization_info': true,
  'footer.page_number': true,
  'cover_page.enabled': true,
  'cover_page.form.name': true,
  'cover_page.form.description': true,
  'cover_page.form.image': true,
  'cover_page.title': true,
  'cover_page.timestamp': true,
  'cover_page.image.enabled': true,
  'cover_page.image.caption': true,
  'cover_page.map.enabled': true,
  'cover_page.map.repeatables': true,
  'cover_page.map.type': 'roadmap',
  'cover_page.map.size': '360x400',
  'cover_page.metadata.enabled': true,
  'cover_page.metadata.created': true,
  'cover_page.metadata.updated': true,
  'cover_page.metadata.status': true,
  'cover_page.metadata.location': true,
  'cover_page.metadata.project': true,
  'cover_page.metadata.assigned': true
};

if (DATA.config.advanced) {
  DATA.config = { ...DATA.config, ...REPORT_CONFIG };
}

const SETTING = (setting, defaultValue) => {
  return DATA.config[setting] != null ? DATA.config[setting] : defaultValue;
};

const { landscape, size } = DATA.config;

const IMAGE_SIZE = () => {
  const data = landscape ? '-ls' : '';
  const default_class = landscape ? 'photo-landscape' : '';

  switch (size) {
    case 'Legal':
      return `photo-legal${data}`;
    case 'Tabloid':
      return `photo-tabloid${data}`;
    case 'Ledger':
      return `photo-ledger${data}`;
    case 'A4':
      return `photo-a4${data}`;
    case 'A3':
      return `photo-a3${data}`;
    default:
      return default_class;
  }
};

const MAP_OPTIONS = (mapSize) => {
  const options = {
    maptype: SETTING('cover_page.map.type'),
    size: mapSize
  };

  return options;
};

const SET_MAP_CLASS = () => {
  if (!SETTING('cover_page.metadata.enabled')) {
    return 'ls-nometa';
  }

  switch (size) {
    case 'Legal':
      return landscape ? 'legal-ls' : '';
    case 'Tabloid':
      return landscape ? 'tabloid-ls' : '';
    case 'A4':
      return landscape ? 'a4-ls' : '';
    case 'Letter':
      return landscape ? 'letter-ls' : '';
    case 'Ledger':
      return landscape ? '' : 'ledger-pt';
    default:
      return '';
  }
};

const SET_MAP_OPTIONS = () => {
  const ledger_options = landscape ? MAP_OPTIONS('400x400') : MAP_OPTIONS('600x400');

  if (!SETTING('cover_page.metadata.enabled')) {
    return MAP_OPTIONS('800x400');
  }

  if (size === 'Ledger') {
    return ledger_options;
  } else if (!landscape) {
    return MAP_OPTIONS('360x400');
  }

  switch (size) {
    case 'Letter':
    case 'A4':
      return MAP_OPTIONS('400x400');
    case 'Legal':
    case 'Tabloid':
      return MAP_OPTIONS('600x400');
    case 'A3':
      return MAP_OPTIONS('800x400');
    default:
      return MAP_OPTIONS('360x400');
  }
};

function initialize() {
  // validate parameters and initialize report data
}

function formatValue(element, value, { defaultFormatter }) {
  // special case logic here
  return defaultFormatter(element, value);
}

Here is an example of what the form's PDF will look like.

Screen Capture of an Example PDF

Screen Capture of an Example PDF