diff --git a/CHANGES.md b/CHANGES.md index e7b1a736..bb8355b0 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -2,6 +2,7 @@ ## Version 1.26 +- Restyle Org Limits - Add new options to hide buttons in popup ## Version 1.25 diff --git a/addon/event-monitor.js b/addon/event-monitor.js index d30221cd..fa1d2ead 100644 --- a/addon/event-monitor.js +++ b/addon/event-monitor.js @@ -53,9 +53,9 @@ class Model { let channel = args.get("channel"); this.selectedChannel = channel; this.selectedChannelType = channel.endsWith("__e") ? "platformEvent" : "standardPlatformEvent"; - } else if(args.get("channelType")){ - this.selectedChannelType = args.get("channelType") - } else{ + } else if (args.get("channelType")){ + this.selectedChannelType = args.get("channelType"); + } else { this.selectedChannelType = channelTypes[0].value; } } @@ -139,7 +139,7 @@ class App extends React.Component { let channels = sessionChannel ? sessionChannel : []; let query; - if(channels.length == 0){ + if (channels.length == 0){ if (channelType == "standardPlatformEvent"){ query = "SELECT Label, QualifiedApiName, DeveloperName FROM EntityDefinition" + " WHERE IsCustomizable = FALSE AND IsEverCreatable = TRUE" @@ -151,18 +151,18 @@ class App extends React.Component { + " AND KeyPrefix LIKE 'e%' ORDER BY Label ASC"; } await sfConn.rest("/services/data/v" + apiVersion + "/tooling/query?q=" + encodeURIComponent(query)) - .then(result => { - result.records.forEach((channel) => { - channels.push({ - name: channel.QualifiedApiName, - label: channel.Label + " (" + channel.QualifiedApiName + ")" + .then(result => { + result.records.forEach((channel) => { + channels.push({ + name: channel.QualifiedApiName, + label: channel.Label + " (" + channel.QualifiedApiName + ")" + }); }); + }) + .catch(err => { + console.error("An error occured fetching Event Channels of type " + channelType + ": ", err.message); }); - }) - .catch(err => { - console.error("An error occured fetching Event Channels of type " + channelType + ": ", err.message); - }); - sessionStorage.setItem(sfHost + "_" + channelType, JSON.stringify(channels)); + sessionStorage.setItem(sfHost + "_" + channelType, JSON.stringify(channels)); } return channels; } @@ -201,25 +201,24 @@ class App extends React.Component { model.selectedChannelType = e.target.value; const urlParams = new URLSearchParams(window.location.search); - urlParams.set('channelType', model.selectedChannelType); - window.history.replaceState(null, '', '?' + urlParams.toString()); + urlParams.set("channelType", model.selectedChannelType); + window.history.replaceState(null, "", "?" + urlParams.toString()); this.getEventChannels(); model.didUpdate(); } onChannelSelection(e) { - let { model } = this.props; + let {model} = this.props; model.selectedChannel = e.target.value; const urlParams = new URLSearchParams(window.location.search); - urlParams.set('channel', model.selectedChannel); - window.history.replaceState(null, '', '?' + urlParams.toString()); + urlParams.set("channel", model.selectedChannel); + window.history.replaceState(null, "", "?" + urlParams.toString()); model.didUpdate(); } - onReplayIdChange(e) { let {model} = this.props; model.replayId = e.target.value; @@ -253,7 +252,7 @@ class App extends React.Component { //Load Salesforce Replay Extension let replayExtension = new cometdReplayExtension(); - replayExtension.setChannel(channelSuffix +model.selectedChannel); + replayExtension.setChannel(channelSuffix + model.selectedChannel); replayExtension.setReplay(model.replayId); replayExtension.setExtensionEnabled = true; cometd.registerExtension("SalesforceReplayExtension", replayExtension); diff --git a/addon/limits.css b/addon/limits.css index 184c39b2..5bbe8647 100644 --- a/addon/limits.css +++ b/addon/limits.css @@ -30,13 +30,13 @@ figcaption > div { display:block; content:""; } -.gauge:before, .meter { - width:10rem; - height:5rem; +.gauge:before, .meter { + width:10rem; + height:5rem; } -.gauge:before { - border-radius:5rem 5rem 0 0; - background:grey; +.gauge:before { + border-radius:5rem 5rem 0 0; + background:grey; } .gauge:after { position:absolute; @@ -86,29 +86,6 @@ body { font-size: 11px; margin: 0; } -.sf-link { - margin-right: 1em; - background-color: rgb(6, 28, 63); - border-radius: 3px; - line-height: 2em; - text-decoration: none; - display: inline-block; - padding: 2px; - color: white; - padding-right: 1em; -} -.sf-link svg { - width: 2em; - height: 2em; - display: block; - margin-left: 1px; - margin-right: 1em; - float: left; - background-color: #ef7ead; - border-radius: 2px; - fill: white; -} - .error-message { font-size: 1.2em; @@ -176,4 +153,9 @@ body { .body.empty { animation: spin 4s linear infinite; } - +#user-info { + font-size: .8125rem; +} +#user-info span { + font-size: 1em; +} \ No newline at end of file diff --git a/addon/limits.html b/addon/limits.html index 5f4405c2..32918e22 100644 --- a/addon/limits.html +++ b/addon/limits.html @@ -1,16 +1,22 @@ - - - ... - - - - -
- - - - - + + + + ... + + + + + + + + +
+ + + + + + \ No newline at end of file diff --git a/addon/limits.js b/addon/limits.js index c4b28d90..472350c0 100644 --- a/addon/limits.js +++ b/addon/limits.js @@ -1,5 +1,6 @@ /* global React ReactDOM */ import {sfConn, apiVersion} from "./inspector.js"; +import {copyToClipboard} from "./data-load.js"; /* global initButton */ class Model { @@ -7,8 +8,17 @@ class Model { this.reactCallback = null; this.sfLink = "https://" + sfHost; this.spinnerCount = 0; + this.title = "Org Limits"; + this.userInfo = "..."; this.allLimitData = []; this.errorMessages = []; + this.sortOptions = [{label: "Consumption", value: "consumption"}, {label: "A-Z %", value: "asc"}]; + this.sortBy = this.sortOptions[1]; + + let userInfoPromise = sfConn.soap(sfConn.wsdl(apiVersion, "Partner"), "getUserInfo", {}); + this.spinFor("userInfo", userInfoPromise, (res) => { + this.userInfo = res.userFullName + " / " + res.userName + " / " + res.organizationName; + }); } didUpdate(cb) { @@ -34,10 +44,6 @@ class Model { .catch(err => console.log("error handling failed", err)); } - title() { - return "Org Limits"; - } - setLimitsData(res) { let self = this; this.allLimitData = []; @@ -48,7 +54,8 @@ class Model { "label": self.humanizeName(key), "description": "...", "max": res[key].Max, - "remaining": res[key].Remaining + "remaining": res[key].Remaining, + "consumption": (res[key].Max - res[key].Remaining) / res[key].Max }); }); } @@ -64,6 +71,10 @@ class Model { this.setLimitsData(res); }); } + + copyAsJson() { + copyToClipboard(JSON.stringify(this.allLimitData ? this.allLimitData : this.allLimitData, null, " "), null, " "); + } } @@ -116,39 +127,79 @@ class LimitData extends React.Component { } class App extends React.Component { + + constructor(props) { + super(props); + this.model = this.props.vm; + this.onCopyAsJson = this.onCopyAsJson.bind(this); + this.onSortBy = this.onSortBy.bind(this); + } + + sortLimits(data, sortBy) { + const sortFunctions = { + consumption: (a, b) => b.consumption - a.consumption, + asc: (a, b) => a.label.localeCompare(b.label) + }; + return data.sort(sortFunctions[sortBy] || (() => 0)); + } + + onCopyAsJson() { + this.model.copyAsJson(); + this.model.didUpdate(); + } + onSortBy(e){ + this.model.sortBy = e.target.value; + this.model.allLimitData = this.sortLimits(this.model.allLimitData, this.model.sortBy); + this.model.didUpdate(); + } + render() { - let vm = this.props.vm; - document.title = vm.title(); - return ( - h("div", {}, - h("div", { - className: "object-bar" - }, - h("img", { - id: "spinner", - src: "", - hidden: vm.spinnerCount == 0 - }), - h("a", { - href: vm.sfLink, - className: "sf-link" - }, - h("svg", { - viewBox: "0 0 24 24" - }, - h("path", { - d: "M18.9 12.3h-1.5v6.6c0 .2-.1.3-.3.3h-3c-.2 0-.3-.1-.3-.3v-5.1h-3.6v5.1c0 .2-.1.3-.3.3h-3c-.2 0-.3-.1-.3-.3v-6.6H5.1c-.1 0-.3-.1-.3-.2s0-.2.1-.3l6.9-7c.1-.1.3-.1.4 0l7 7v.3c0 .1-.2.2-.3.2z" - }) + let model = this.props.vm; + document.title = model.title; + return h("div", {}, + h("div", {id: "user-info"}, + h("a", {href: `https://${sfConn.instanceHostname}`, className: "sf-link"}, + h("svg", {viewBox: "0 0 24 24"}, + h("path", {d: "M18.9 12.3h-1.5v6.6c0 .2-.1.3-.3.3h-3c-.2 0-.3-.1-.3-.3v-5.1h-3.6v5.1c0 .2-.1.3-.3.3h-3c-.2 0-.3-.1-.3-.3v-6.6H5.1c-.1 0-.3-.1-.3-.2s0-.2.1-.3l6.9-7c.1-.1.3-.1.4 0l7 7v.3c0 .1-.2.2-.3.2z"}) + ), + " Salesforce Home" ), - " Salesforce Home" - ) + h("h1", {}, model.title), + h("span", {}, " / " + model.userInfo), + h("div", {className: "flex-right"}, + h("a", {href: "https://developer.salesforce.com/docs/atlas.en-us.api_rest.meta/api_rest/resources_limits.htm", target: "_blank", id: "help-btn", title: "Org Limits Help", onClick: null}, + h("div", {className: "icon"}) + ), + h("div", {id: "spinner", role: "status", className: "slds-spinner slds-spinner_small slds-spinner_inline", hidden: model.spinnerCount == 0}, + h("span", {className: "slds-assistive-text"}), + h("div", {className: "slds-spinner__dot-a"}), + h("div", {className: "slds-spinner__dot-b"}), + ), ), - h("div", { - className: "body" - }, - vm.allLimitData.map(limitData => - h(LimitData, limitData) - ) + ), + h("div", {className: "area", id: "result-area"}, + h("div", {className: "result-bar"}, + h("h1", {}, "Limits"), + h("div", {className: "button-group"}, + h("button", {disabled: model.allLimitData.length == 0, onClick: this.onCopyAsJson, title: "Copy raw JSON to clipboard"}, "Copy") + ), + h("div", {className: "flex-right"}, + h("select", {value: model.sortBy, onChange: this.onSortBy, className: ""}, + h("option", {value: "none", disabled: true, defaultValue: true, hidden: true}, "Sort By"), + model.sortOptions.map(opt => h("option", {key: opt.value, value: opt.value}, opt.label)) + ), + ), + ), + h("div", {id: "result-table", ref: "scroller"}, + h("div", {}, + h("div", { + className: "body" + }, + model.allLimitData.map(limitData => + h(LimitData, limitData) + ) + ) + ) ) ) );