Form Schema
Print the Form Schema (Simple)
Using this code in your BODY and STYLES 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.
<h1><%= form.name %></h1>
<br>
<div>
<% RENDER(record, null, ({element, value, renderSection, renderRepeatableItems, container, index, feature, parent, allValues}) => { %>
<% if (element.isSectionElement) { %>
<div class='field-section'>
<h1 class='field-section-title'><%= element.label %></h1>
<div class='field-section-content'>
<% renderSection() %>
</div>
</div>
<% } else { %>
<div class='field'>
<h2 class='field-label'><%= element.label %></h2>
<div class='field-value pre'>Required?: <%= element._isRequired %></div>
</div>
<% } %>
<% }) %>
</div>
Leave Blank
Leave Blank
h1, h2, h3, h4, h5, h6 {
margin: 0;
padding: 0;
}
.flex-align-center {
display: flex;
align-items: center;
}
.field {
padding: 5px 0;
border-bottom: 1px solid #e1e1e1;
page-break-inside: avoid;
display: flex;
}
.photo-field {
display: block;
page-break-inside: auto;
}
.field:last-of-type {
border-bottom: none;
}
.meta-field {
border-bottom: 1px solid #ddd;
margin-bottom: 5px;
padding-bottom: 5px;
}
.meta-field-group .meta-field:first-of-type {
border-bottom: none;
}
.field-label {
font-weight: 900;
width: 30%;
font-size: 12px;
box-sizing: border-box;
}
.meta-field-label {
box-sizing: border-box;
color: #41403b;
font-size: 13px;
font-weight: 900;
margin-bottom: 10px;
text-transform: uppercase;
width: 30%;
}
.field-label-full {
font-weight: normal;
width: 100%;
box-sizing: border-box;
font-size: 12px;
}
.field-value {
width: 70%;
box-sizing: border-box;
border-left: 1px solid #e1e1e1;
padding-left: 10px;
}
.meta-field-value {
color: #7d7d7d;
padding-bottom: 3px;
font-size: 14px;
}
.meta-field-value.flex-align-center svg {
padding-right: 5px;
}
.meta-field-value-large {
font-size: 18px;
}
.cover-photo-container, .photo-column figure {
position: relative;
}
.cover-photo-container img {
height: 300px;
object-fit: cover;
}
.cover-photo-container figcaption, .photo-column figcaption {
position: absolute;
bottom: 18px;
margin: 0;
padding: 10px;
background: rgba(255,255,255,.85);
border-radius: 0 7px 7px 0;
border: 1px dashed #bbb;
border-left: none;
margin-right: 15px;
left: 0;
}
.cover-photo-container img.photo-landscape { height: 500px; }
.cover-photo-container img.photo-legal { height: 560px; }
.cover-photo-container img.photo-legal-ls { height: 498px; }
.cover-photo-container img.photo-tabloid { height: 740px; }
.cover-photo-container img.photo-tabloid-ls { height: 740px; }
.cover-photo-container img.photo-ledger { height: 740px; }
.cover-photo-container img.photo-ledger-ls { height: 740px; }
.cover-photo-container img.photo-a4 { height: 380px; }
.cover-photo-container img.photo-a4-ls { height: 470px; }
.cover-photo-container img.photo-a3 { height: 840px; }
.cover-photo-container img.photo-a3-ls { height: 790px; }
.pre {
white-space: pre-wrap;
}
.page-break-before {
page-break-before: always;
}
.page-break-after {
page-break-after: always;
}
.field-value .photo {
max-width: 470px;
max-height: 470px;
margin-top: 6px;
}
.field-value .signature {
max-width: 470px;
max-height: 470px;
margin-top: 6px;
}
.field-value .status-color {
width: 14px;
height: 14px;
border-radius: 3px;
margin-right: 8px;
position: relative;
float: left;
top: 0px;
}
.meta-field-value .status-color {
width: 14px;
height: 14px;
border-radius: 3px;
margin-right: 8px;
position: relative;
float: left;
top: 0px;
}
.field-section {
border-bottom: 2px solid #e1e1e1;
border-right: 2px solid #e1e1e1;
margin-bottom: 8px;
}
.field-section.metadata {
color: #41403b;
margin: 10px 0;
border: none;
}
.field-section-title {
background: #f4f4f4;
font-weight: bold;
font-size: 18px;
border-left: 3px solid #000;
margin-top: 20px;
margin-bottom: 8px;
padding: 7px 12px;
}
.field-repeatable-title {
background: #f4f4f4;
font-weight: bold;
font-size: 18px;
border-left: 3px solid #000;
margin-top: 20px;
margin-bottom: 8px;
padding: 7px 12px;
}
.field-repeatable-item {
padding: 10px 0;
}
.field-repeatable-item .field-section-title,
.field-repeatable-item .field-repeatable-title {
font-size: 16px;
border-color: #ddd;
background: #f9f9f9;
}
.field-repeatable-item > h2 {
background: #f9f9f9;
border-left: 3px solid #ddd;
border-bottom: 2px solid #ddd;
font-size: 16px;
padding: 7px 12px;
margin: 10px 0;
}
.field-repeatable-table {
border-collapse: collapse;
width: 100%;
margin: 10px 0;
}
.field-repeatable-table thead {
display: table-header-group;
break-inside: avoid;
}
.field-repeatable-table td, .field-repeatable-table th {
border: 1px solid #ccc;
padding: 4px;
text-align: left;
}
.field-repeatable-table th {
background: #eee;
}
.field-repeatable-table tr {
page-break-inside: avoid;
}
.field-repeatable-table tr:nth-child(even) {
background-color: #f4f4f4;
}
.photo-row {
padding-top: 4px;
}
.photo-column {
box-sizing: border-box;
display: inline;
}
.photo-column figure {
margin-inline-start: 0px;
margin-inline-end: 0px;
margin-block-start: 0px;
margin-block-end: 0px;
display: inline;
}
.photo-column figure img {
max-height: 460px;
max-width: 49%;
}
.photo-column figure img.photo-landscape { max-height: 760px; max-width: 47%; }
.photo-column figure img.photo-legal-ls { max-height: 680px; }
.photo-column figure img.photo-tabloid { max-height: 960px; }
.photo-column figure img.photo-tabloid-ls { max-height: 960px; }
.photo-column figure img.photo-ledger { max-height: 960px; }
.photo-column figure img.photo-ledger-ls { max-height: 740px; }
.photo-column figure img.photo-a4-ls { max-height: 760px; }
.photo-column figure img.photo-a3 { max-height: 960px; }
.photo-column figure img.photo-a3-ls { max-height: 960px; }
.footer {
display: flex;
justify-content: space-between;
color: #7d7d7d;
}
.footer-logo {
display: flex;
align-items: center;
}
.footer-logo img, .footer-branding img {
height: 40px;
padding-right: 10px;
}
.footer-company-info, .footer-meta, .footer-branding {
display: flex;
align-items: center;
}
.footer-company {
display: flex;
align-items: center;
color: #41403b;
}
.footer-meta {
text-align: center;
}
.footer-address, .footer-meta {
font-size: 11px;
}
.page-info {
width: 300px;
float: right;
text-align: right;
margin-top: 15px;
}
.title {
padding: 15px 0px 0px 0px;
page-break-inside: avoid;
display: flex;
padding-right: 5px;
}
.title hr {
margin: 6px 0px 12px;
display: block;
height: 1px;
border: 0;
border-top: 1px solid #ddd;
padding: 0;
}
.title h1 {
color: #41403b;
font-size: 3em;
font-weight: 300;
}
.title h2 {
font-weight: 400;
color: #7d7d7d;
font-size: 14px;
}
.title-label {
flex: 4 1;
}
.title-image {
flex: 1 1;
text-align: right;
}
.title-image img {
width: 100px;
border-radius: 5px;
border: 10px solid #fff;
box-shadow: 0 0 3px #ccc;
}
.meta-wrapper {
display: flex;
align-items: flex-start;
}
.meta-map {
margin-top: 20px;
}
.meta-map img {
width: auto;
height: 400px;
}
.meta-map img.ls-nometa {
width: 100%;
height: auto;
}
.meta-map img.letter-ls { height: 440px; }
.meta-map img.legal-ls { height: 440px; }
.meta-map img.tabloid-ls { height: 440px; }
.meta-map img.ledger-pt { height: 440px; }
.meta-map img.a4-ls { height: 440px; }
.meta-map .field {
border: none;
padding-top: 0;
}
.meta-fields-container {
display: flex;
flex-direction: column;
flex: 1;
margin-left: 20px;
}
.meta-fields {
flex: 1;
margin-bottom: 8px;
padding-bottom: 8px;
border-bottom: 1px solid #f4f4f4;
}
.meta-fields:last-of-type {
border-bottom: none;
}
.header {
display: flex;
justify-content: space-between;
margin-top: 15px;
color: #7d7d7d;
}
.header-title, .header-record {
align-items: center;
font-style: italic;
}
Leave Blank
Here is an example of what the form's PDF will resemble.
Print the Form 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>Read-Only</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.
Updated 7 months ago