python/whylogs/viz/html/js/whylogs-script.js
"use strict";
(function () {
const MESSAGES = {
error: {
noInputElementFound: "It seems we could not find the file input element.",
noBrowserSupport: "This browser does not seem to support the `files` property of file inputs.",
noFileSelected: "Please select a file.",
noProfileSelected: "Please select a profile.",
fileAPINotSupported: "The file API is not supported on this browser yet.",
invalidJSONFile:
"The JSON file you are trying to load does not match the expected whylogs format. Please check the file and try again.",
},
};
// HTML Elements
const $selectedReferenceProfile = $(".wl__selected-reference-profile");
const $selectedProfile = $(".wl__selected-profile");
const $removeReferenceProfileButton = $("#remove-reference-profile-button")
const $featureFileName = $(".wl__feature-file-name")
const $notifCircleContainer = $(".notif-circle-container")
const $boxes = $('input[name=checkbox]:checked');
const $closeIcon = $("#close-icon");
const $openSignUpText = $("#open-sign-up-text")
const $signUpText = $(".sign-up-text");
const $dropdownArrowIcon = $("#dropdown-arrow-icon");
const $referenceJsonForm = $("#reference-json-form");
const $referencefileInput = $("#reference-file-input");
const $compareProfile = $("#compare-profile");
const $featureCount = $(".wl__feature-count");
const $sidebarFeatureNameList = $(".wl__sidebar-feature-name-list");
const $featureCountDiscrete = $(".wl__feature-count--discrete");
const $featureCountNonDiscrete = $(".wl__feature-count--non-discrete");
const $featureCountUnknown = $(".wl__feature-count--unknown");
const $tableBody = $(".wl__table-body");
const $featureSearch = $("#wl__feature-search");
const $featureFilterInput = $(".wl__feature-filter-input");
const $jsonForm = $("#json-form");
const $fileInput = $("#file-input");
const $tableContent = $("#table-content");
const $tableMessage = $("#table-message");
const $sidebarContent = $("#sidebar-content");
const $multiProfileWrap = $("#sidebar-content-multi-profile");
const $propertyPanelTitle = $(".wl-property-panel__title");
const $propertyPanelProfileName = $(".wl-property-panel__table-th-profile");
const $filterOptions = $(".filter-options");
// Constants and variables
let featureSearchValue = "";
let isActiveInferredType = {};
let propertyPanelData = [];
let referencePropertyPanelData = [];
let jsonData = {};
let referenceJsonData = {};
let dataForRead = {};
let featureDataForTableForAllProfiles = {};
let numOfProfilesBasedOnType = {};
let selectedProfiles = [];
selectedProfiles.push("0");
// Util functions
function debounce(func, wait, immediate) {
let timeout;
return function () {
const context = this;
const args = arguments;
const later = function () {
timeout = null;
if (!immediate) func.apply(context, args);
};
const callNow = immediate && !timeout;
clearTimeout(timeout);
timeout = setTimeout(later, wait);
if (callNow) func.apply(context, args);
};
}
function handleSearch() {
const tableBodyChildrens = $tableBody.children();
const featureListChildren = $sidebarFeatureNameList.children();
let featureCount = 0;
for (let i = 0; i < tableBodyChildrens.length; i++) {
const name = tableBodyChildrens[i].dataset.featureName.toLowerCase();
const type = tableBodyChildrens[i].dataset.inferredType.toLowerCase();
if (isActiveInferredType[type] && name.startsWith(featureSearchValue)) {
tableBodyChildrens[i].style.display = "";
featureCount++;
} else {
tableBodyChildrens[i].style.display = "none";
}
}
if (jsonData) {
for (let i = 0; i < featureListChildren.length; i++) {
const name = featureListChildren[i].dataset.featureName.toLowerCase();
const type = featureListChildren[i].dataset.inferredType.toLowerCase();
if (isActiveInferredType[type] && name.startsWith(featureSearchValue)) {
featureListChildren[i].style.display = "";
} else {
featureListChildren[i].style.display = "none";
}
}
}
$featureCount.html(featureCount);
}
function fixNumberTo(number, decimals = 3) {
return parseFloat(number).toFixed(decimals);
}
function getQuantileValues(data) {
return {
min: fixNumberTo(data[0]),
firstQuantile: fixNumberTo(data[3]),
median: fixNumberTo(data[4]),
thirdQuantile: fixNumberTo(data[5]),
max: fixNumberTo(data[8]),
};
}
function formatLabelDate(timestamp) {
const date = new Date(timestamp);
const format = d3.timeFormat("%Y-%m-%d %I:%M:%S %p %Z");
return format(date);
}
function scrollToFeatureName(event) {
const TABLE_HEADER_OFFSET = 90;
const $tableWrap = $(".wl-table-wrap");
const featureNameId = event.currentTarget.dataset.featureNameId;
const currentOffsetTop = $tableWrap.scrollTop();
const offsetTop = $('[data-scroll-to-feature-name="' + featureNameId + '"]').offset().top;
$tableWrap.animate(
{
scrollTop: currentOffsetTop + offsetTop - TABLE_HEADER_OFFSET,
},
500,
);
}
const chipElement = (chip) => `<span class="wl-table-cell__bedge">${chip}</span>`;
const chipElementTableData = (value) => `<td class="wl-property-panel__table-td" >${chipElement(value)}</td>`;
const chartBoxElement = (chartTitle, chart) => `
<div class="chart-box-wrap mb-4">
<div class="chart-box" id="chart-box">
<div class="chart-box-title display-flex">${chartTitle}</div>
<div class="svg-container">${chart}</div>
</div>
</div>
`
const frequentItemBoxElement = (chartTitle, items) => `
<div class="frequent-item-box-wrap mb-4">
<div class="frequent-item-box display-flex" id="chart-box">
<tbody class="wl-property-panel__frequent-items">
${items}
</tbody>
</div>
</div>
`
const colorsForDistingushingCharts = (color, text) => `
<div class="colors-for-distingushing-charts">
<div class="circle-color" style="background: ${color};"></div>
<text alignment-baseline="middle" style="font-size: 15px;">${text}</text>
</div>
`
class GenerateChartParams {
constructor(height, width, data, bottomMargin=20, topMargin=5) {
this.MARGIN = {
TOP: topMargin,
RIGHT: 5,
BOTTOM: bottomMargin,
LEFT: 55,
};
this.SVG_WIDTH = width;
this.SVG_HEIGHT = height;
this.CHART_WIDTH = this.SVG_WIDTH - this.MARGIN.LEFT - this.MARGIN.RIGHT;
this.CHART_HEIGHT = this.SVG_HEIGHT - this.MARGIN.TOP - this.MARGIN.BOTTOM;
this.svgEl = d3.create("svg").attr("width", this.SVG_WIDTH).attr("height", this.SVG_HEIGHT);
this.maxYValue = d3.max(data, (d) => Math.abs(d.axisY));
this.xScale = d3
.scaleBand()
.domain(data.map((d) => d.axisX))
.range([this.MARGIN.LEFT, this.MARGIN.LEFT + this.CHART_WIDTH]);
this.yScale = d3
.scaleLinear()
.domain([0, this.maxYValue * 1.02])
.range([this.CHART_HEIGHT, 0]);
}
}
const referenceProfilePanelHeight = () => {
const pageHeight = $(document).height() - 48;
if ($(".clickable-test-feature-wrap").height() <= pageHeight) {
$(".clickable-test-feature-wrap").css("height", pageHeight)
} else {
$(".clickable-test-feature-wrap").css("height", "auto")
}
}
function getGraphHtml(data, height = 75, width = 350, index = 0, referenceProfile = false, propertyPanelGraph = false) {
const sizes = new GenerateChartParams(height, width, data, 5)
let {
MARGIN,
CHART_HEIGHT,
svgEl,
maxYValue,
xScale,
yScale
} = sizes
const color = ["#369BAC", '#2683C9']
if (propertyPanelGraph) {
svgEl = d3.create("svg").attr("width", width).attr("height", height);
}
// Add the y Axis
if (!referenceProfile) {
svgEl
.append("g")
.attr("transform", "translate(" + MARGIN.LEFT + ", " + MARGIN.TOP + ")")
.call(d3.axisLeft(yScale).tickValues([0, maxYValue/2, maxYValue]));
}
const gChart = svgEl.append("g");
gChart
.selectAll(".bar")
.data(data)
.enter()
.append("rect")
.classed("bar", true)
.attr("width", xScale.bandwidth() - 1)
.attr("height", (d) => CHART_HEIGHT - yScale(d.axisY))
.attr("x", (d) => xScale(d.axisX))
.attr("y", (d) => yScale(d.axisY) + MARGIN.TOP)
.attr("fill", color[index]);
return svgEl._groups[0][0].outerHTML;
}
function generateDoubleHistogramChart(currentWidth, histogramData, overlappedHistogramData) {
let yFormat,
xFormat;
const sizes = new GenerateChartParams(230, currentWidth, histogramData)
let {
MARGIN,
SVG_WIDTH,
SVG_HEIGHT,
CHART_WIDTH,
CHART_HEIGHT,
svgEl,
xScale,
yScale
} = sizes
svgEl = d3.create("svg")
.attr("preserveAspectRatio", "xMinYMin meet")
.attr("viewBox", "0 0 600 400")
.classed("svg-content-responsive", true)
const xAxis = d3.axisBottom(xScale).ticks(SVG_WIDTH / 80, xFormat).tickSizeOuter(0);
const yAxis = d3.axisLeft(yScale).ticks(CHART_HEIGHT / 40, yFormat);
yFormat = yScale.tickFormat(100, yFormat);
svgEl.append("g")
.attr("transform", `translate(${MARGIN.LEFT}, ${MARGIN.TOP})`)
.call(yAxis)
.call(g => g.select(".domain").remove())
.call(g => g.selectAll(".tick line")
.attr("x2", CHART_WIDTH)
.attr("stroke-opacity", 0.1))
.call(g => g.append("text")
.attr("x", -MARGIN.LEFT)
.attr("y", 10)
.attr("fill", "currentColor")
.attr("text-anchor", "start"));
svgEl.append("g").append("g")
.attr("transform", `translate(0,${SVG_HEIGHT - MARGIN.BOTTOM})`)
.call(xAxis)
.call(g => g.select(".domain").remove())
.call(g => g.selectAll(".tick line").remove())
.call(g => g.append("text")
.attr("x", SVG_WIDTH - MARGIN.RIGHT)
.attr("y", 27)
.attr("fill", "currentColor")
.attr("text-anchor", "end"));
const gChart = svgEl.append("g");
gChart
.selectAll(".bar")
.data(histogramData)
.enter()
.append("rect")
.classed("bar", true)
.attr("width", xScale.bandwidth())
.attr("height", (d) => CHART_HEIGHT - yScale(d.axisY))
.attr("x", (d) => xScale(d.axisX))
.attr("y", (d) => yScale(d.axisY) + MARGIN.TOP)
.attr("fill", "#369BAC")
.style("opacity","0.6");
const gChart1 = svgEl.append("g");
gChart1
.selectAll(".bar")
.data(overlappedHistogramData)
.enter()
.append("rect")
.classed("bar", true)
.attr("width", xScale.bandwidth())
.attr("height", (d) => CHART_HEIGHT - yScale(d.axisY))
.attr("x", (d) => xScale(d.axisX))
.attr("y", (d) => yScale(d.axisY) + MARGIN.TOP)
.attr("fill", "#2683C9")
.style("opacity", "0.6");
return svgEl._groups[0][0].outerHTML;
}
function generateBarChart(currentWidth, histogramData, overlappedHistogramData) {
let yFormat,
xFormat;
const data = histogramData.map((profile, index) => {
return {
group: index,
profile: profile.axisY,
reference_profile: overlappedHistogramData[index].axisY
}
}).slice(0, 20)
const sizes = new GenerateChartParams(230, currentWidth, histogramData, undefined, 1)
let {
MARGIN,
SVG_WIDTH,
SVG_HEIGHT,
CHART_WIDTH,
CHART_HEIGHT,
svgEl,
xScale,
yScale
} = sizes
svgEl = d3.create("svg")
.attr("preserveAspectRatio", "xMinYMin meet")
.attr("viewBox", "0 0 600 400")
.classed("svg-content-responsive", true)
const subgroups = ['profile', 'reference_profile']
xScale.padding([0.3])
const xAxis = d3.axisBottom(xScale).ticks(SVG_WIDTH / 80, xFormat).tickSizeOuter(0);
const yAxis = d3.axisLeft(yScale).ticks(SVG_HEIGHT / 40, yFormat);
yFormat = yScale.tickFormat(100, yFormat);
svgEl.append("g")
.attr("transform", `translate(${MARGIN.LEFT}, ${MARGIN.TOP})`)
.call(yAxis)
.call(g => g.select(".domain").remove())
.call(g => g.selectAll(".tick line")
.attr("x2", CHART_WIDTH)
.attr("stroke-opacity", 0.1))
.call(g => g.append("text")
.attr("x", -MARGIN.LEFT)
.attr("y", 10)
.attr("fill", "currentColor")
.attr("text-anchor", "start"));
svgEl.append("g")
.attr("transform", `translate(0,${SVG_HEIGHT - MARGIN.BOTTOM})`)
.call(xAxis)
.call(g => g.select(".domain").remove())
.call(g => g.selectAll(".tick line").remove())
.call(g => g.append("text")
.attr("x", SVG_WIDTH - MARGIN.RIGHT)
.attr("y", 27)
.attr("fill", "currentColor")
.attr("text-anchor", "end"));
// Another scale for subgroup position?
const xSubgroup = d3.scaleBand()
.domain(subgroups)
.range([0, xScale.bandwidth()])
// color palette = one color per subgroup
const color = d3.scaleOrdinal()
.domain(subgroups)
.range(['#369BAC', '#2683C9'])
svgEl.append("g")
.selectAll("g")
// Enter in data = loop group per group
.data(data)
.enter()
.append("g")
.attr("transform", function(d) { return "translate(" + xScale(d.group) + ",0)"; })
.selectAll("rect")
.data(function(d) { return subgroups.map(function(key) { return {key: key, value: d[key]}; }); })
.enter().append("rect")
.attr("x", function(d) { return xSubgroup(d.key); })
.attr("y", function(d) { return yScale(d.value); })
.attr("width", xSubgroup.bandwidth())
.attr("height", function(d) { return (CHART_HEIGHT - yScale(d.value)); })
.attr("fill", function(d) { return color(d.key); })
.style("opacity", "0.6");
return svgEl._groups[0][0].outerHTML;
}
function generatePositiveNegativeChart(currentWidth, histogramData, overlappedHistogramData) {
const data = histogramData.map((value, index) => {
const difference = value.axisY - overlappedHistogramData[index].axisY
const negativeValues = difference < 0 ? difference : 0
return [+value.axisY, negativeValues]
}).flat().slice(0, 20)
let yFormat,
xFormat;
const sizes = new GenerateChartParams(230, currentWidth, histogramData, undefined, 1)
let {
MARGIN,
SVG_WIDTH,
SVG_HEIGHT,
CHART_WIDTH,
CHART_HEIGHT,
svgEl
} = sizes
svgEl = d3.create("svg")
.attr("preserveAspectRatio", "xMinYMin meet")
.attr("viewBox", "0 0 600 400")
.classed("svg-content-responsive", true)
const y0 = Math.max(Math.abs(d3.min(data)), Math.abs(d3.max(data)));
const yScale = d3.scaleLinear()
.domain([-y0 * 1.02, y0 * 1.02])
.range([CHART_HEIGHT,0])
const xScale = d3.scaleBand()
.domain(d3.range(data.length)) // so that chart's height has 102% height of the maximum value
.rangeRound([MARGIN.LEFT, SVG_WIDTH])
.padding([0.1]);
const xAxis = d3.axisBottom(xScale).ticks(SVG_WIDTH / 80, xFormat).tickSizeOuter(0);
const yAxis = d3.axisLeft(yScale).ticks(CHART_HEIGHT / 40, yFormat);
yFormat = yScale.tickFormat(100, yFormat);
svgEl.append("g")
.attr("transform", `translate(${MARGIN.LEFT}, ${MARGIN.TOP})`)
.call(yAxis)
.call(g => g.select(".domain").remove())
.call(g => g.selectAll(".tick line")
.attr("x2", CHART_WIDTH )
.attr("stroke-opacity", 0.1))
.call(g => g.append("text")
.attr("x", - MARGIN.LEFT)
.attr("y", 10)
.attr("fill", "currentColor")
.attr("text-anchor", "start"));
svgEl
.append("g")
.attr("transform", `translate(0,${SVG_HEIGHT - MARGIN.BOTTOM})`)
.call(xAxis)
.call(g => g.select(".domain").remove())
.call(g => g.selectAll(".tick>line").remove())
.call(g => g.append("text")
.attr("x", SVG_WIDTH - MARGIN.RIGHT)
.attr("y", 27)
.attr("fill", "currentColor")
.attr("text-anchor", "end"));
svgEl.selectAll(".bar")
.data(data)
.enter().append("rect")
.attr("class", function(d) { return d < 0 ? "bar negative" : "bar positive"; })
.attr("y", function(d) { return yScale(Math.max(0, d)); })
.attr("x", function(d, i) { return xScale(i); })
.attr("height", function(d) { return Math.abs(yScale(d) - yScale(0)); })
.attr("width", xScale.bandwidth());
return svgEl._groups[0][0].outerHTML;
}
function numericalDriftChart(getDoubleHistogramChart) {
let colorsForDistingushingChartHTMLElement = '';
colorsForDistingushingChartHTMLElement +=`
<p>Data Distribution Chart</p>
<div class="display-flex">
${colorsForDistingushingCharts("#369BAC", "Current")}
${colorsForDistingushingCharts("#2683C9", "Reference")}
</div>
`
$(".clickable-test-feature-body").html(`
${chartBoxElement(colorsForDistingushingChartHTMLElement, getDoubleHistogramChart)}
`);
}
function categoricalDriftChart(getBarChart, getPositiveNegative) {
$(".clickable-test-feature-body").html(`
${chartBoxElement('<p>Bar Chart</p>', getBarChart)}
${chartBoxElement('<p>Difference Bar Chart</p>', getPositiveNegative)}
`);
}
function sortWithIndeces(toSort) {
for (let i = 0; i < toSort.length; i++) {
toSort[i] = [toSort[i], i];
}
toSort.sort((left, right) => {
return right[0] < left[0] ? -1 : 1;
});
toSort.sortIndices = [];
for (let j = 0; j < toSort.length; j++) {
toSort.sortIndices.push(toSort[j][1]);
toSort[j] = toSort[j][0];
}
return toSort;
}
const getProfileCharts = (key, getDoubleHistogramChart, getBarChart, getPositiveNegative) => {
if (jsonData.columns[key].numberSummary.isDiscrete) {
$("#page-button").text("Categorical Data")
categoricalDriftChart(getBarChart, getPositiveNegative)
} else {
$("#page-button").text("Numerical Data")
numericalDriftChart(getDoubleHistogramChart)
}
}
function openReferencePropertyPanel(referenceItems, items, profileItems, key, chart, getDoubleHistogramChart, getBarChart, getPositiveNegative) {
const chartInfoItem = (drift, driftName) => `
<div class="info">
<div>${drift}</div>
<p>${driftName}</p>
</div>
`
const $clickableTestFeatureWrap = $(".clickable-test-feature-wrap");
const $pagesButtons = $(".page-button");
const $pagesButton = $pagesButtons[0];
let chipString = "",
frequentItemString = "",
referenceFrequentItemString = "";
$pagesButtons.removeClass("activ-pages-button")
$($pagesButton).addClass("activ-pages-button")
$tableContent.addClass("d-none")
$clickableTestFeatureWrap.removeClass("d-none")
// frequentItemString += `${items.forEach((item) => {chipElementTableData(item.value)})}`
const sortedItems = items.map((item) => +Object.values(item)[0])
// .sort((a,b) => +b.value - (+a.value))
sortWithIndeces(sortedItems).sortIndices.forEach(
(item) => {
frequentItemString += `
${frequentItemBoxElement('',chipElementTableData(items[item].value))}
`
referenceFrequentItemString += `
${frequentItemBoxElement('',chipElementTableData(referenceItems[item].value))}
`
}
);
$("#page-button").on("click", function () {
getProfileCharts(
key,
getDoubleHistogramChart,
getBarChart,
getPositiveNegative
)
$(".clickable-test-feature-body").removeClass("d-none");
$(".frequent-items-body").html(``);
referenceProfilePanelHeight()
})
getProfileCharts(
key,
getDoubleHistogramChart,
getBarChart,
getPositiveNegative
)
$(".clickable-test-feature-body").removeClass("d-none");
$(".frequent-items-body").html(``);
$("#frequent-item-button").on("click", function () {
$(".frequent-items-body").html(`
<div class="display-flex">
<div class="fequent-items-wrap">
<div class="chart-box-title frequent-item-box-to-title display-flex">
items
</div>
${frequentItemString}
</div>
<div class="fequent-items-wrap">
<div class="chart-box-title frequent-item-box-to-title display-flex">
reference profile items
</div>
${referenceFrequentItemString}
</div>
</div>
`);
$(".clickable-test-feature-body").addClass("d-none");
referenceProfilePanelHeight()
})
$("#chart").html(chart);
chipString += `
${chartInfoItem(profileItems.numberSummary.count.toString(), "Total Count")}
${chartInfoItem(fixNumberTo(profileItems.numberSummary.mean), "Mean")}
`
$(".chart-info").html(chipString);
referenceProfilePanelHeight()
}
function openProfilePropertyPanel(items, infType, chart) {
$("#wl-property-panel__chart").html(chart);
let chipString = "";
const chipElementEstimation = (count) =>
`<td class="wl-property-panel__table-td wl-property-panel__table-td-profile" >${count}</td>`;
items.forEach((item) => {
chipString += `
<tr class="wl-property-panel__table-tr">
${chipElementTableData(item.value)}
${chipElementEstimation(item.count)}
</tr>
`;
});
$(".wl-property-panel__frequent-items").html(chipString);
if (infType === "non-discrete") {
$propertyPanelTitle.html("Histogram data:");
$propertyPanelProfileName.html("Bin values");
} else if (infType === "discrete") {
$propertyPanelTitle.html("Frequent items:");
$propertyPanelProfileName.html("Counts");
}
$(".wl-property-panel").addClass("wl-property-panel--open");
$(".wl-table-wrap").addClass("wl-table-wrap--narrow");
}
function openPropertyPanel(referenceItems, items, infType, feature) {
let getGraph = null,
getPropertyPanelGraph = null,
getDoubleHistogramChart,
getBarChart,
getPositiveNegative,
currentWidth = 600,
propertyPanelGraph = true;
if (referencePropertyPanelData[feature[0]][0]) {
getDoubleHistogramChart = generateDoubleHistogramChart(currentWidth, feature[1].chartData[0], feature[1].chartData[1])
getBarChart = generateBarChart(currentWidth, feature[1].chartData[0], feature[1].chartData[1])
getPositiveNegative = generatePositiveNegativeChart(currentWidth, feature[1].chartData[0], feature[1].chartData[1])
items = referencePropertyPanelData[feature[0]][0]
getGraph = getGraphHtml(feature[1].chartData[1], 50, 280, 0, true, propertyPanelGraph)
}
getPropertyPanelGraph = getPropertyPanelGraphHtml(jsonData.columns[feature[0]], feature[0])
if (jsonData) {
if (items.length > 0 && items !== "undefined") {
if (referencePropertyPanelData[feature[0]][0]) {
openReferencePropertyPanel(
referenceItems,
items,
referenceJsonData.columns[feature[0]],
feature[0],
getGraph,
getDoubleHistogramChart,
getBarChart,
getPositiveNegative
)
} else {
openProfilePropertyPanel(items, infType, getPropertyPanelGraph)
}
} else {
}
}
}
function handleClosePropertyPanel() {
$(".wl-property-panel").removeClass("wl-property-panel--open");
$(".wl-table-wrap").removeClass("wl-table-wrap--narrow");
$(".wl-property-panel__frequent-items").html("");
}
function getPropertyPanelGraphHtml (column) {
let chartString = "";
const freqData = [];
const histData = [];
const freqChart = (chart) =>
`<div class="wl-property-panel__chart--single"><div class="wl-property-panel__chart-title">Frequent Items Data</div>${chart}</div>`;
const histChart = (chart) =>
`<div class="wl-property-panel__chart--single"><div class="wl-property-panel__chart-title">Histogram Data</div>${chart}</div>`;
if (column.numberSummary) {
if (column.frequentItems && column.frequentItems.items) {
column.frequentItems.items.forEach((item, index) => {
freqData.push({
axisY: item.estimate,
axisX: index,
});
});
}
if (column.numberSummary.histogram && column.numberSummary.histogram.counts) {
column.numberSummary.histogram.counts.slice(0, 30).forEach((count, index) => {
histData.push({
axisY: count,
axisX: index,
});
});
}
if (column.numberSummary.isDiscrete) {
if (freqData.length > 0) chartString += freqChart(getGraphHtml(freqData, 130));
if (histData.length > 0) chartString += histChart(getGraphHtml(histData, 130));
} else {
if (histData.length > 0) chartString += histChart(getGraphHtml(histData, 130));
if (freqData.length > 0) chartString += freqChart(getGraphHtml(freqData, 130));
}
}
return chartString;
}
function sidebarContentHeight() {
const $sidebarContentPadding = +$("#sidebar-content-single-profile").css("padding").replace('px','') * 2
const $sidebarContentHeight = $("#sidebar-content-single-profile").height() + $sidebarContentPadding
const $sidebar = $(".sidebar")
$sidebar.css("margin-bottom", `${$sidebarContentHeight}px`)
}
function checkCurrentProfile(item, referenceItem) {
if (referenceJsonData && Object.values(referenceJsonData)) {
return referenceItem
} else {
return item
}
}
// Override and populate HTML element values
function updateHtmlElementValues() {
$notifCircleContainer.addClass("d-none")
$dropdownArrowIcon.css("transform","rotate(180deg)")
$filterOptions.removeClass("d-none");
checkCurrentProfile(true, false) ?
$("#dif-from-ref").addClass("d-none"):
$("#dif-from-ref").removeClass("d-none")
sidebarContentHeight()
$sidebarFeatureNameList.html("");
$tableMessage.addClass("d-none");
Object.entries(featureDataForTableForAllProfiles).forEach((feature) => {
// strings for tableToShow
let tempChartDataString = "";
let referenceTempChartDataString = "";
feature[1].chartData.forEach((chartData, index) => {
if (selectedProfiles.includes(String(index))) {
tempChartDataString += `<div>${
chartData.length > 0
? getGraphHtml(feature[1].chartData[0])
: '<span class="wl-table-cell__bedge-wrap">No data to show the chart</span>'
}</div>`;
if (feature[1].chartData[1] && feature[1].chartData[1].length > 0) {
referenceTempChartDataString+= `<div>${
chartData.length > 0
? getGraphHtml(feature[1].chartData[1], ...[,,], 1, true)
: '<span class="wl-table-cell__bedge-wrap">No data to show the chart</span>'
}</div>`;
}
}
});
let diffFromRef = "";
feature[1].frequentItemsElemString.forEach((frequentItemElemString, index) => {
if (selectedProfiles.includes(String(index))) {
diffFromRef += ` <div class="wl-table-cell__bedge-wrap">${Math.floor(Math.random() * 10)}</div>`;
}
});
let freaquentItemsElmString = "";
feature[1].frequentItemsElemString.forEach((frequentItemElemString, index) => {
if (selectedProfiles.includes(String(index))) {
freaquentItemsElmString += ` <div class="wl-table-cell__bedge-wrap text-align-center">${frequentItemElemString}</div>`;
}
});
let inferredTypeString = "";
feature[1].inferredType.forEach((inferredType, index) => {
if (selectedProfiles.includes(String(index))) {
inferredType ? (inferredTypeString += `<div>${inferredType}</div>`) : (inferredTypeString += `<div>-</div>`);
}
});
let totalCountString = "";
feature[1].totalCount.forEach((totalCount, index) => {
if (selectedProfiles.includes(String(index))) {
totalCount ? (totalCountString += `<div>${totalCount}</div>`) : (totalCountString += `<div>-</div>`);
}
});
let nullRationString = "";
feature[1].nullRatio.forEach((nullRatio, index) => {
if (selectedProfiles.includes(String(index))) {
nullRatio ? (nullRationString += `<div> ${nullRatio}</div>`) : (nullRationString += `<div>-</div>`);
}
});
let estUniqueValString = "";
feature[1].estUniqueVal.map((estUniqueVal, index) => {
if (selectedProfiles.includes(String(index))) {
estUniqueVal ? (estUniqueValString += `<div>${estUniqueVal}</div>`) : (estUniqueValString += `<div>$-</div>`);
}
});
let meanString = "";
feature[1].mean.forEach((mean, index) => {
if (selectedProfiles.includes(String(index))) {
mean ? (meanString += `<div>${mean}</div>`) : (meanString += `<div>-</div>`);
}
});
let stddevString = "";
feature[1].stddev.forEach((stddev, index) => {
if (selectedProfiles.includes(String(index))) {
stddev ? (stddevString += `<div>${stddev}</div>`) : (stddevString += `<div>-</div>`);
}
});
let dataTypeString = "";
feature[1].dataType.forEach((dataType, index) => {
if (selectedProfiles.includes(String(index))) {
dataType ? (dataTypeString += `<div>${dataType}</div>`) : (dataTypeString += `<div>-</div>`);
}
});
let dataTypeCountString = "";
feature[1].dataTypeCount.forEach((dataTypeCount, index) => {
if (selectedProfiles.includes(String(index))) {
dataTypeCount
? (dataTypeCountString += `<div>${dataTypeCount}</div>`)
: (dataTypeCountString += `<div>-</div>`);
}
});
let quantilesMinString = "";
feature[1].quantiles.forEach((quantiles, index) => {
if (selectedProfiles.includes(String(index))) {
quantiles.min
? (quantilesMinString += `<div>${quantiles.min}</div>`)
: (quantilesMinString += `<div>-</div>`);
}
});
let quantilesMedianString = "";
feature[1].quantiles.forEach((quantiles, index) => {
if (selectedProfiles.includes(String(index))) {
quantiles.median
? (quantilesMedianString += `<div>${quantiles.median}</div>`)
: (quantilesMedianString += `<div>-</div>`);
}
});
let quantilesThirdQuantileString = "";
feature[1].quantiles.forEach((quentiles, index) => {
if (selectedProfiles.includes(String(index))) {
quentiles.thirdQuantile
? (quantilesThirdQuantileString += `<div>${quentiles.thirdQuantile}</div>`)
: (quantilesThirdQuantileString += `<div>-</div>`);
}
});
let quantilesMaxString = "";
feature[1].quantiles.forEach((quantiles, index) => {
if (selectedProfiles.includes(String(index))) {
quantiles.max
? (quantilesMaxString += `<div>${quantiles.max}</div>`)
: (quantilesMaxString += `<div>-</div>`);
}
});
let quantilesFirsString = "";
feature[1].quantiles.forEach((quantiles, index) => {
if (selectedProfiles.includes(String(index))) {
quantiles.firstQuantile
? (quantilesFirsString += `<div>${quantiles.firstQuantile}</div>`)
: (quantilesFirsString += `<div>-</div>`);
}
});
const showButton = selectedProfiles.some(
(profile) => profile !== null && feature[1].inferredType[profile].toLowerCase() !== "unknown",
);
const $tableRow = $(
`
<li class="wl-table-row${showButton ? " wl-table-row--clickable" : ""}" data-feature-name="${
feature[0]
}" data-inferred-type="${
feature[1].inferredType[parseInt(selectedProfiles[0] || "0")]
}" data-scroll-to-feature-name="${feature[0]}" style="display: none;">
<div class="wl-table-cell">
<div class="wl-table-cell__title-wrap">
<h4 class="wl-table-cell__title">${feature[0]}</h4>
</div>
<div class="wl-table-cell__graph-wrap">
<div class="display-flex">` +
tempChartDataString +
referenceTempChartDataString +
`</div>
</div></div>
<div class="wl-table-cell wl-table-cell--top-spacing align-middle ${checkCurrentProfile(`d-none`, ``)}" style="max-width: 270px; padding-right: 18px">` +
checkCurrentProfile(``, diffFromRef) +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle">` +
freaquentItemsElmString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle">` +
inferredTypeString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
totalCountString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
nullRationString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
estUniqueValString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle">` +
dataTypeString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle">` +
dataTypeCountString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
meanString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
stddevString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
quantilesMinString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
quantilesFirsString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
quantilesMedianString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
quantilesThirdQuantileString +
`</div><div class="wl-table-cell wl-table-cell--top-spacing align-middle text-end">` +
quantilesMaxString +
`</div></li>`,
);
if (jsonData) {
const $tableRowButton = $(`<button class="wl-table-cell__title-button" type="button">View details</button>`);
$tableRowButton.on(
"click",
openPropertyPanel.bind(
this,
referencePropertyPanelData[feature[0]][0],
propertyPanelData[feature[0]][0],
feature[1].inferredType[0].toLowerCase(),
feature
),
);
$tableRow.find(".wl-table-cell__title-wrap").append($tableRowButton);
}
// Update data table rows/columns
$tableBody.append($tableRow);
$sidebarFeatureNameList.append(
`
<li id="filter-list-item" class="wl_filter-list-item list-group-item js-list-group-item" data-feature-name="${feature[0]}" data-inferred-type="${feature[1].inferredType}">
<div class="arrow-icon-container">
<div class="wl_list-item-dot"></div>
<img
class="d-none wl_arrow-icon"
src=""
/>
</div>
<span data-feature-name-id="${feature[0]}" >${feature[0]}</span>
</li>
`,
);
});
if (jsonData) {
const countDiscrete = Object.values(numOfProfilesBasedOnType).reduce(
(acc, feature) => (acc += feature.discrete.length),
0,
);
const countNonDiscrete = Object.values(numOfProfilesBasedOnType).reduce(
(acc, feature) => (acc += feature.nonDiscrete.length),
0,
);
const countUnknown = Object.values(numOfProfilesBasedOnType).reduce(
(acc, feature) => (acc += feature.unknown.length),
0,
);
$featureCountDiscrete.html(countDiscrete);
$featureCountNonDiscrete.html(countNonDiscrete);
$featureCountUnknown.html(countUnknown);
$selectedProfile.html(formatLabelDate(+dataForRead.properties[0].dataTimestamp));
if (checkCurrentProfile(false, true) && dataForRead.properties[1]) {
$selectedReferenceProfile.html(formatLabelDate(+dataForRead.properties[1].dataTimestamp))
}
}
}
function renderList() {
const tableBodyChildrens = $tableBody.children();
const featureListChildren = $sidebarFeatureNameList.children();
let featureCount = 0;
$.each($featureFilterInput, (_, filterInput) => {
isActiveInferredType[filterInput.value.toLowerCase()] = filterInput.checked;
});
for (let i = 0; i < tableBodyChildrens.length; i++) {
const type = tableBodyChildrens[i].dataset.inferredType.toLowerCase();
if (isActiveInferredType[type]) {
tableBodyChildrens[i].style.display = "";
featureCount++;
}
}
for (let i = 0; i < featureListChildren.length; i++) {
const type = featureListChildren[i].dataset.inferredType.toLowerCase();
if (isActiveInferredType[type]) {
featureListChildren[i].style.display = "";
}
}
$featureCount.html(featureCount);
}
function getDataForRead(dataForRead, jsonData, referneceData) {
if (!dataForRead.properties) {
dataForRead.properties = [];
}
dataForRead.properties.push(jsonData.properties);
Object.entries(jsonData.columns).forEach((feature) => {
let tempFeatureName = feature[0];
if (!dataForRead.columns) {
dataForRead.columns = [];
}
if (!dataForRead.columns[tempFeatureName]) dataForRead.columns[tempFeatureName] = [];
dataForRead.columns[tempFeatureName].push(feature[1]);
if (
referneceData &&
referneceData.columns[tempFeatureName].numberSummary
) {
dataForRead.properties.push(referenceJsonData.properties)
const {
numberSummary,
frequentItems,
...referenceDataForRead
} = dataForRead.columns[tempFeatureName][0]
dataForRead.columns[tempFeatureName].push({
...referenceDataForRead,
referenceNumberSummary: referneceData.columns[tempFeatureName].numberSummary,
referenceFrequentItems: referneceData.columns[tempFeatureName].frequentItems
})
}
});
return dataForRead
}
function mapProfileDataToReadData(jsonData, dataForRead, referneceData) {
try {
dataForRead = getDataForRead(dataForRead, jsonData, referneceData)
} catch (e) {
alert("You selected wrong reference profile. Please select again.")
removeReferenceProfile()
}
makeFeatureDataForAllProfilesToShowOnTable(
featureDataForTableForAllProfiles,
dataForRead,
numOfProfilesBasedOnType,
);
}
function makeFeatureDataForAllProfilesToShowOnTable(
featureDataForTableForAllProfiles,
dataForRead,
numOfProfilesBasedOnType,
) {
if (Object.keys(featureDataForTableForAllProfiles).length !== 0) {
featureDataForTableForAllProfiles = {};
}
if (Object.keys(numOfProfilesBasedOnType).length !== 0) {
numOfProfilesBasedOnType = {};
}
Object.entries(dataForRead.columns).map((feature) => {
if (!featureDataForTableForAllProfiles[feature[0]]) {
featureDataForTableForAllProfiles[feature[0]] = {
totalCount: [],
inferredType: [],
nullRatio: [],
estUniqueVal: [],
dataType: [],
dataTypeCount: [],
quantiles: [],
mean: [],
stddev: [],
chartData: [],
refereneChartData: [],
frequentItemsElemString: [],
};
numOfProfilesBasedOnType[feature[0]] = {
discrete: [],
nonDiscrete: [],
unknown: [],
};
propertyPanelData[feature[0]] = [];
referencePropertyPanelData[feature[0]] = [];
}
let iteration = 0;
feature[1].forEach((tempFeatureValues) => {
if (tempFeatureValues.numberSummary) {
let inffTypeTemp = tempFeatureValues.numberSummary.isDiscrete ? "Discrete" : "Non-Discrete";
tempFeatureValues.numberSummary.isDiscrete
? numOfProfilesBasedOnType[feature[0]].discrete.push(feature[0])
: numOfProfilesBasedOnType[feature[0]].nonDiscrete.push(feature[0]);
featureDataForTableForAllProfiles[feature[0]].totalCount.push(tempFeatureValues.numberSummary.count);
featureDataForTableForAllProfiles[feature[0]].inferredType.push(inffTypeTemp);
featureDataForTableForAllProfiles[feature[0]].quantiles.push(
getQuantileValues(tempFeatureValues.numberSummary.quantiles.quantileValues),
);
featureDataForTableForAllProfiles[feature[0]].mean.push(fixNumberTo(tempFeatureValues.numberSummary.mean));
featureDataForTableForAllProfiles[feature[0]].stddev.push(
fixNumberTo(tempFeatureValues.numberSummary.stddev),
);
} else {
numOfProfilesBasedOnType[feature[0]].unknown.push(feature[0]);
featureDataForTableForAllProfiles[feature[0]].inferredType.push("Unknown");
featureDataForTableForAllProfiles[feature[0]].totalCount.push("-");
let quantiles = {
min: "-",
firstQuantile: "-",
median: "-",
thirdQuantile: "-",
max: "-",
};
featureDataForTableForAllProfiles[feature[0]].quantiles.push(quantiles);
featureDataForTableForAllProfiles[feature[0]].mean.push("-");
featureDataForTableForAllProfiles[feature[0]].stddev.push("-");
}
featureDataForTableForAllProfiles[feature[0]].estUniqueVal.push(
feature[1].uniqueCount ? fixNumberTo(tempFeatureValues.uniqueCount.estimate) : "-",
);
featureDataForTableForAllProfiles[feature[0]].nullRatio.push(
tempFeatureValues.schema.typeCounts.NULL ? tempFeatureValues.schema.typeCounts.NULL : "0",
);
featureDataForTableForAllProfiles[feature[0]].dataType.push(tempFeatureValues.schema.inferredType.type);
featureDataForTableForAllProfiles[feature[0]].dataTypeCount.push(
tempFeatureValues.schema.typeCounts[featureDataForTableForAllProfiles[feature[0]].dataType],
);
featureDataForTableForAllProfiles[feature[0]].chartData[iteration] = [];
featureDataForTableForAllProfiles[feature[0]].frequentItemsElemString[iteration] = "";
if (
featureDataForTableForAllProfiles[feature[0]].inferredType[iteration] === "Discrete" &&
tempFeatureValues.frequentItems &&
tempFeatureValues.frequentItems.items.length
) {
// Chart
tempFeatureValues.frequentItems.items.forEach((item, index) => {
featureDataForTableForAllProfiles[feature[0]].chartData[iteration].push({
axisY: item.estimate,
axisX: index,
});
});
if (tempFeatureValues.referenceFrequentItems) {
tempFeatureValues.referenceFrequentItems.items.forEach((item, index) => {
featureDataForTableForAllProfiles[feature[0]].chartData[1].push({
axisY: item.estimate,
axisX: index,
});
});
}
// Frequent item chips / bedge
propertyPanelData[feature[0]][iteration] = tempFeatureValues.frequentItems.items.reduce((acc, item) => {
acc.push({
value: item.jsonValue,
count: item.estimate,
});
return acc;
}, []);
const slicedFrequentItems = tempFeatureValues.frequentItems.items.slice(0, 5);
for (let fi = 0; fi < slicedFrequentItems.length; fi++) {
featureDataForTableForAllProfiles[feature[0]].frequentItemsElemString[iteration] +=
'<span class="wl-table-cell__bedge">' + slicedFrequentItems[fi].jsonValue + "</span>";
}
} else {
if (featureDataForTableForAllProfiles[feature[0]].inferredType[iteration] === "Non-Discrete") {
// Chart
if (tempFeatureValues.numberSummary) {
// Histogram chips / bedge
propertyPanelData[feature[0]][0] = tempFeatureValues.numberSummary.histogram.counts.reduce(
(acc, value, index) => {
acc.push({
value: value,
count: tempFeatureValues.numberSummary.histogram.bins[index],
});
return acc;
},
[],
);
tempFeatureValues.numberSummary.histogram.counts.slice(0, 30).forEach((count, index) => {
featureDataForTableForAllProfiles[feature[0]].chartData[0].push({
axisY: count,
axisX: index,
});
});
}
// Frequent item chips / bedge
featureDataForTableForAllProfiles[feature[0]].frequentItemsElemString[iteration] = "No data to show";
} else {
featureDataForTableForAllProfiles[feature[0]].frequentItemsElemString[iteration] = "No data to show";
featureDataForTableForAllProfiles[feature[0]].chartData[iteration] = [];
propertyPanelData[feature[0]] = [];
}
if (tempFeatureValues.referenceNumberSummary) {
referencePropertyPanelData[feature[0]][0] = tempFeatureValues.referenceNumberSummary.histogram.counts.reduce(
(acc, value, index) => {
acc.push({
value: value,
count: tempFeatureValues.referenceNumberSummary.histogram.bins[index],
});
return acc;
},
[],
);
tempFeatureValues.referenceNumberSummary.histogram.counts.slice(0, 30).forEach((count, index) => {
featureDataForTableForAllProfiles[feature[0]].chartData[1].push({
axisY: count,
axisX: index,
});
});
}
}
if(tempFeatureValues.referenceFrequentItems &&
featureDataForTableForAllProfiles[feature[0]].inferredType[0] === "Discrete"
){
featureDataForTableForAllProfiles[feature[0]].chartData[1] = []
tempFeatureValues.referenceFrequentItems.items.forEach((item, index) => {
featureDataForTableForAllProfiles[feature[0]].chartData[1].push({
axisY: item.estimate,
axisX: index,
});
});
}
iteration += 1;
});
});
$featureCount.html(Object.values(featureDataForTableForAllProfiles).length);
}
function updateTableMessage(message) {
$tableMessage.find("p").html(message);
}
function compareArrays(array, target) {
if (array.length !== target.length) return false;
array.sort();
target.sort();
for (let i = 0; i < array.length; i++) {
if (array[i] !== target[i]) {
return false;
}
}
return true;
}
function checkJSONValidityForSingleProfile(data) {
const VALID_KEY_NAMES = ["properties", "columns"];
const keys = Object.keys(data);
return keys.length === 2 && compareArrays(VALID_KEY_NAMES, keys);
}
function hasFalseValue(array) {
return array.some((obj) => checkJSONValidityForSingleProfile(obj) === false);
}
function checkJSONValidityForMultiProfile(data) {
const values = Object.values(data);
if (!values && !values.length) return false;
if (hasFalseValue(values)) return false;
return true;
}
function showDataVisibility() {
$tableMessage.addClass("d-none");
$sidebarContent.removeClass("d-none");
$tableContent.removeClass("d-none");
}
function hideDataVisibility() {
$tableMessage.removeClass("d-none");
$sidebarContent.addClass("d-none");
$tableContent.addClass("d-none");
$compareProfile.removeClass("d-none");
}
function addFileName() {
const fileName = $fileInput.val().split('\\').pop();
if (fileName) {
$featureFileName.html(fileName.split('.json'))
}
}
function removeReferenceProfile() {
referenceJsonData = undefined
dataForRead = {};
featureDataForTableForAllProfiles = {};
numOfProfilesBasedOnType = {};
$(".reference-table-head").addClass("d-none")
mapProfileDataToReadData(jsonData, dataForRead, referenceJsonData);
$tableBody.html("");
updateHtmlElementValues();
renderList();
$(".wl-selected-profile").addClass("d-none")
$(".wl-compare-profile").removeClass("d-none")
}
function loadFile(inputId, jsonForm) {
isActiveInferredType = {};
$featureFilterInput.html("");
$sidebarFeatureNameList.html("");
$tableBody.html("");
$(".form-check-input").prop("checked", true);
addFileName()
if (typeof window.FileReader !== "function") {
updateTableMessage(MESSAGES.error.fileAPINotSupported);
return;
}
dataForRead = {};
featureDataForTableForAllProfiles = {};
numOfProfilesBasedOnType = {};
handleClosePropertyPanel();
const input = document.getElementById(inputId);
if (!input) {
updateTableMessage(MESSAGES.error.noInputElementFound);
} else if (!input.files) {
updateTableMessage(MESSAGES.error.noBrowserSupport);
} else if (!input.files[0]) {
updateTableMessage(MESSAGES.error.noFileSelected);
} else {
const file = input.files[0];
const fr = new FileReader();
fr.onload = (e) => receivedText(e, jsonForm, inputId);
fr.readAsText(file);
}
}
function receivedText(e, jsonForm, inputId) {
const lines = e.target.result;
let data = JSON.parse(lines);
if (inputId === "file-input") {
if (checkJSONValidityForMultiProfile(data)) {
jsonData = Object.values(data)[0]
} else {
jsonData = data;
}
removeReferenceProfile()
$(".reference-table-head").addClass("d-none")
$(".wl-selected-profile").addClass("d-none")
$compareProfile.removeClass("d-none");
} else {
if (checkJSONValidityForMultiProfile(data)) {
referenceJsonData = Object.values(data)[0]
} else {
referenceJsonData = data;
}
$(".reference-table-head").removeClass("d-none")
$(".wl-selected-profile").removeClass("d-none")
$(".wl-compare-profile").addClass("d-none")
}
mapProfileDataToReadData(jsonData, dataForRead, referenceJsonData);
if (data) {
$(".compare-select").addClass("d-none");
$multiProfileWrap.removeClass("d-none");
selectedProfiles[0] = "0";
$tableBody.html("");
updateHtmlElementValues();
renderList();
showDataVisibility();
jsonForm.trigger("reset");
} else {
$tableBody.html("");
updateTableMessage(MESSAGES.error.invalidJSONFile);
hideDataVisibility();
return;
}
}
function checkedBoxes() {
const item = Object.values($boxes).find(
function(value) {
return $('#' + $(value)[0].id).is(":checked")
}
);
if (item) {
$notifCircleContainer.removeClass("d-none")
}
}
$removeReferenceProfileButton.on("click", removeReferenceProfile)
$(document).on("click", ".page-button", function(e) {
const $pagesButtons = $(".page-button"),
$pagesButtonIndex = $pagesButtons.index(e.target),
$pagesButton = $(".page-button")[$pagesButtonIndex]
$pagesButtons.removeClass("activ-pages-button")
$($pagesButton).addClass("activ-pages-button")
})
$(window).ready(function() {
sidebarContentHeight()
})
$("#property-panel-close-icon").on("click", function (e) {
const $clickableTestFeatureWrap = $(".clickable-test-feature-wrap")
$tableContent.removeClass("d-none")
$clickableTestFeatureWrap.addClass("d-none")
});
$dropdownArrowIcon.on("click", function () {
const filterClass = $filterOptions.attr("class");
if (filterClass.indexOf("d-none") > 0) {
$notifCircleContainer.addClass("d-none")
$filterOptions.removeClass("d-none");
$("#dropdown-container").css("background-color", '#FFF')
$dropdownArrowIcon.css("transform","rotate(180deg)")
} else {
$filterOptions.addClass("d-none");
checkedBoxes()
$("#dropdown-container").css("background-color", 'none')
$dropdownArrowIcon.css("transform","rotate(0)")
}
});
$closeIcon.on("click", function () {
$signUpText.addClass("d-none");
sidebarContentHeight()
$(".open-sign-up-text-notif-container").removeClass("d-none")
});
$openSignUpText.on("click", function () {
$signUpText.removeClass("d-none");
sidebarContentHeight()
$(".open-sign-up-text-notif-container").addClass("d-none")
});
$(document).on("click", ".js-list-group-item span", function (e) {
const listItem = $("li>span"),
listItemIndex = listItem.index(e.target),
$listItemDot = $(".wl_list-item-dot")[listItemIndex],
$arrowIcon = $(".wl_arrow-icon")[listItemIndex]
listItem.css("padding-left", "15px")
$(".wl_list-item-dot").removeClass("d-none")
$(".wl_arrow-icon").addClass("d-none")
$($arrowIcon).removeClass("d-none")
$($listItemDot).addClass("d-none")
$(listItem[listItemIndex]).css("padding-left", "8px")
});
// Bind event listeners
$fileInput.on("change", () => loadFile("file-input", $jsonForm));
$referencefileInput.on("change", () => loadFile("reference-file-input", $referenceJsonForm));
$(document).on("click", ".js-list-group-item span", scrollToFeatureName);
$featureSearch.on(
"input",
debounce((event) => {
featureSearchValue = event.target.value.toLowerCase();
handleSearch();
}, 300),
);
$featureFilterInput.on("change", (event) => {
const filterType = event.target.value.toLowerCase();
const isChecked = event.target.checked;
isActiveInferredType[filterType] = isChecked;
handleSearch();
});
$(".wl-property-panel__button").on("click", handleClosePropertyPanel);
})();