diff --git a/.release-version b/.release-version index a9184766b..faf0dcbb0 100644 --- a/.release-version +++ b/.release-version @@ -1 +1 @@ -3.43.0 +3.44.0 diff --git a/Gemfile.lock b/Gemfile.lock index 5961b4f20..90a991f86 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -28,70 +28,71 @@ GIT GEM remote: https://rubygems.org/ specs: - actioncable (7.1.1) - actionpack (= 7.1.1) - activesupport (= 7.1.1) + actioncable (7.1.3) + actionpack (= 7.1.3) + activesupport (= 7.1.3) nio4r (~> 2.0) websocket-driver (>= 0.6.1) zeitwerk (~> 2.6) - actionmailbox (7.1.1) - actionpack (= 7.1.1) - activejob (= 7.1.1) - activerecord (= 7.1.1) - activestorage (= 7.1.1) - activesupport (= 7.1.1) + actionmailbox (7.1.3) + actionpack (= 7.1.3) + activejob (= 7.1.3) + activerecord (= 7.1.3) + activestorage (= 7.1.3) + activesupport (= 7.1.3) mail (>= 2.7.1) net-imap net-pop net-smtp - actionmailer (7.1.1) - actionpack (= 7.1.1) - actionview (= 7.1.1) - activejob (= 7.1.1) - activesupport (= 7.1.1) + actionmailer (7.1.3) + actionpack (= 7.1.3) + actionview (= 7.1.3) + activejob (= 7.1.3) + activesupport (= 7.1.3) mail (~> 2.5, >= 2.5.4) net-imap net-pop net-smtp rails-dom-testing (~> 2.2) - actionpack (7.1.1) - actionview (= 7.1.1) - activesupport (= 7.1.1) + actionpack (7.1.3) + actionview (= 7.1.3) + activesupport (= 7.1.3) nokogiri (>= 1.8.5) + racc rack (>= 2.2.4) rack-session (>= 1.0.1) rack-test (>= 0.6.3) rails-dom-testing (~> 2.2) rails-html-sanitizer (~> 1.6) - actiontext (7.1.1) - actionpack (= 7.1.1) - activerecord (= 7.1.1) - activestorage (= 7.1.1) - activesupport (= 7.1.1) + actiontext (7.1.3) + actionpack (= 7.1.3) + activerecord (= 7.1.3) + activestorage (= 7.1.3) + activesupport (= 7.1.3) globalid (>= 0.6.0) nokogiri (>= 1.8.5) - actionview (7.1.1) - activesupport (= 7.1.1) + actionview (7.1.3) + activesupport (= 7.1.3) builder (~> 3.1) erubi (~> 1.11) rails-dom-testing (~> 2.2) rails-html-sanitizer (~> 1.6) - activejob (7.1.1) - activesupport (= 7.1.1) + activejob (7.1.3) + activesupport (= 7.1.3) globalid (>= 0.3.6) - activemodel (7.1.1) - activesupport (= 7.1.1) - activerecord (7.1.1) - activemodel (= 7.1.1) - activesupport (= 7.1.1) + activemodel (7.1.3) + activesupport (= 7.1.3) + activerecord (7.1.3) + activemodel (= 7.1.3) + activesupport (= 7.1.3) timeout (>= 0.4.0) - activestorage (7.1.1) - actionpack (= 7.1.1) - activejob (= 7.1.1) - activerecord (= 7.1.1) - activesupport (= 7.1.1) + activestorage (7.1.3) + actionpack (= 7.1.3) + activejob (= 7.1.3) + activerecord (= 7.1.3) + activesupport (= 7.1.3) marcel (~> 1.0) - activesupport (7.1.1) + activesupport (7.1.3) base64 bigdecimal concurrent-ruby (~> 1.0, >= 1.0.2) @@ -101,15 +102,15 @@ GEM minitest (>= 5.1) mutex_m tzinfo (~> 2.0) - addressable (2.8.5) + addressable (2.8.6) public_suffix (>= 2.0.2, < 6.0) ast (2.4.2) - autoprefixer-rails (10.4.15.0) + autoprefixer-rails (10.4.16.0) execjs (~> 2) - base64 (0.1.1) - bigdecimal (3.1.4) + base64 (0.2.0) + bigdecimal (3.1.6) bindex (0.8.1) - bootsnap (1.16.0) + bootsnap (1.17.1) msgpack (~> 1.2) bootstrap (4.6.2) autoprefixer-rails (>= 9.1.0) @@ -130,22 +131,22 @@ GEM capybara selenium-webdriver coderay (1.1.3) - concurrent-ruby (1.2.2) + concurrent-ruby (1.2.3) connection_pool (2.4.1) crack (0.4.5) rexml crass (1.0.6) - date (3.3.3) + date (3.3.4) diff-lcs (1.5.0) docile (1.4.0) - drb (2.1.1) + drb (2.2.0) ruby2_keywords erubi (1.12.0) exception_notification (4.5.0) actionmailer (>= 5.2, < 8) activesupport (>= 5.2, < 8) execjs (2.9.1) - factory_bot (6.3.0) + factory_bot (6.4.5) activesupport (>= 5.0.0) faraday (1.10.3) faraday-em_http (~> 1.0) @@ -190,24 +191,24 @@ GEM guard (~> 2.1) guard-compat (~> 1.1) rspec (>= 2.99.0, < 4.0) - hashdiff (1.0.1) + hashdiff (1.1.0) hashie (5.0.0) i18n (1.14.1) concurrent-ruby (~> 1.0) inline_svg (1.9.0) activesupport (>= 3.0) nokogiri (>= 1.6) - io-console (0.6.0) - irb (1.8.3) + io-console (0.7.2) + irb (1.11.1) rdoc - reline (>= 0.3.8) + reline (>= 0.4.2) jquery-rails (4.6.0) rails-dom-testing (>= 1, < 3) railties (>= 4.2.0) thor (>= 0.14, < 2.0) jquery-ui-rails (6.0.1) railties (>= 3.2.16) - json (2.6.3) + json (2.7.1) language_server-protocol (3.17.0.3) launchy (2.5.2) addressable (~> 2.8) @@ -217,7 +218,7 @@ GEM loofah (2.22.0) crass (~> 1.0.2) nokogiri (>= 1.12.0) - lumberjack (1.2.9) + lumberjack (1.2.10) mail (2.8.1) mini_mime (>= 0.1.1) net-imap @@ -228,31 +229,32 @@ GEM method_source (1.0.0) mini_mime (1.1.5) mini_portile2 (2.8.5) - minitest (5.20.0) + minitest (5.21.2) msgpack (1.7.2) multi_json (1.15.0) multipart-post (2.3.0) - mutex_m (0.1.2) + mutex_m (0.2.0) nenv (0.3.0) - net-imap (0.4.2) + net-imap (0.4.9.1) date net-protocol net-pop (0.1.2) net-protocol - net-protocol (0.2.1) + net-protocol (0.2.2) timeout - net-smtp (0.4.0) + net-smtp (0.4.0.1) net-protocol - nio4r (2.5.9) - nokogiri (1.15.5) + nio4r (2.7.0) + nokogiri (1.16.0) mini_portile2 (~> 2.8.2) racc (~> 1.4) notiffany (0.1.3) nenv (~> 0.1) shellany (~> 0.0) - oj (3.16.1) - parallel (1.23.0) - parser (3.2.2.4) + oj (3.16.3) + bigdecimal (>= 3.0) + parallel (1.24.0) + parser (3.3.0.4) ast (~> 2.4.1) racc popper_js (1.16.1) @@ -262,10 +264,10 @@ GEM pry-byebug (3.10.1) byebug (~> 11.0) pry (>= 0.13, < 0.15) - psych (5.1.1.1) + psych (5.1.2) stringio - public_suffix (5.0.3) - puma (6.4.0) + public_suffix (5.0.4) + puma (6.4.2) nio4r (~> 2.0) racc (1.7.3) rack (3.0.8) @@ -280,20 +282,20 @@ GEM rackup (2.1.0) rack (>= 3) webrick (~> 1.8) - rails (7.1.1) - actioncable (= 7.1.1) - actionmailbox (= 7.1.1) - actionmailer (= 7.1.1) - actionpack (= 7.1.1) - actiontext (= 7.1.1) - actionview (= 7.1.1) - activejob (= 7.1.1) - activemodel (= 7.1.1) - activerecord (= 7.1.1) - activestorage (= 7.1.1) - activesupport (= 7.1.1) + rails (7.1.3) + actioncable (= 7.1.3) + actionmailbox (= 7.1.3) + actionmailer (= 7.1.3) + actionpack (= 7.1.3) + actiontext (= 7.1.3) + actionview (= 7.1.3) + activejob (= 7.1.3) + activemodel (= 7.1.3) + activerecord (= 7.1.3) + activestorage (= 7.1.3) + activesupport (= 7.1.3) bundler (>= 1.15.0) - railties (= 7.1.1) + railties (= 7.1.3) rails-controller-testing (1.0.5) actionpack (>= 5.0.1.rc1) actionview (>= 5.0.1.rc1) @@ -305,9 +307,9 @@ GEM rails-html-sanitizer (1.6.0) loofah (~> 2.21) nokogiri (~> 1.14) - railties (7.1.1) - actionpack (= 7.1.1) - activesupport (= 7.1.1) + railties (7.1.3) + actionpack (= 7.1.3) + activesupport (= 7.1.3) irb rackup (>= 1.0.0) rake (>= 12.2) @@ -318,10 +320,10 @@ GEM rb-fsevent (0.11.2) rb-inotify (0.10.1) ffi (~> 1.0) - rdoc (6.5.0) + rdoc (6.6.2) psych (>= 4.0.0) - regexp_parser (2.8.2) - reline (0.3.9) + regexp_parser (2.9.0) + reline (0.4.2) io-console (~> 0.5) rexml (3.2.6) rspec (3.12.0) @@ -346,29 +348,29 @@ GEM rspec-mocks (~> 3.12) rspec-support (~> 3.12) rspec-support (3.12.1) - rubocop (1.57.1) - base64 (~> 0.1.1) + rubocop (1.60.1) json (~> 2.3) language_server-protocol (>= 3.17.0) parallel (~> 1.10) - parser (>= 3.2.2.4) + parser (>= 3.3.0.2) rainbow (>= 2.2.2, < 4.0) regexp_parser (>= 1.8, < 3.0) rexml (>= 3.2.5, < 4.0) - rubocop-ast (>= 1.28.1, < 2.0) + rubocop-ast (>= 1.30.0, < 2.0) ruby-progressbar (~> 1.7) unicode-display_width (>= 2.4.0, < 3.0) rubocop-ast (1.30.0) parser (>= 3.2.1.0) - rubocop-performance (1.19.1) - rubocop (>= 1.7.0, < 2.0) - rubocop-ast (>= 0.4.0) - rubocop-rails (2.21.2) + rubocop-performance (1.20.2) + rubocop (>= 1.48.1, < 2.0) + rubocop-ast (>= 1.30.0, < 2.0) + rubocop-rails (2.23.1) activesupport (>= 4.2.0) rack (>= 1.1) rubocop (>= 1.33.0, < 2.0) + rubocop-ast (>= 1.30.0, < 2.0) ruby-progressbar (1.13.0) - ruby-units (4.0.0) + ruby-units (4.0.1) ruby2_keywords (0.0.5) rubyzip (2.3.2) sass-rails (6.0.0) @@ -382,7 +384,7 @@ GEM sprockets-rails tilt select2-rails (4.0.13) - selenium-webdriver (4.14.0) + selenium-webdriver (4.16.0) rexml (~> 3.2, >= 3.2.5) rubyzip (>= 1.2.2, < 3.0) websocket (~> 1.0) @@ -409,10 +411,10 @@ GEM activesupport (>= 5.2) sprockets (>= 3.0.0) state_machines (0.6.0) - stringio (3.0.8) + stringio (3.1.0) thor (1.3.0) tilt (2.3.0) - timeout (0.4.0) + timeout (0.4.1) tzinfo (2.0.6) concurrent-ruby (~> 1.0) uglifier (4.2.0) diff --git a/app/assets/stylesheets/limber/icons.scss b/app/assets/stylesheets/limber/icons.scss index 1c477d831..d1502d731 100644 --- a/app/assets/stylesheets/limber/icons.scss +++ b/app/assets/stylesheets/limber/icons.scss @@ -4,6 +4,8 @@ // // Where {name} is the name of the icon and {color} is the bootstrap theme color // (e.g. icon-user-primary, icon-user-secondary, icon-user-success, etc.) +// Bootstrap theme colors are: +// primary, secondary, success, danger, warning, info, light, dark $icon-size: 24px; @@ -48,10 +50,24 @@ $icon-size: 24px; // iterate over bootstrap theme colors // icons by Heroicons https://heroicons.com/ - // user-circle + // user-circle (solid) @include icon( 'user', '', $theme-color ); + + // magnifying-glass (solid) + @include icon( + 'search', + '', + $theme-color + ); + + // arrow-left (solid) + @include icon( + 'arrowleft', + '', + $theme-color + ); } diff --git a/app/assets/stylesheets/limber/pipeline-graph.scss b/app/assets/stylesheets/limber/pipeline-graph.scss index 51b6fea9e..f83b5e313 100644 --- a/app/assets/stylesheets/limber/pipeline-graph.scss +++ b/app/assets/stylesheets/limber/pipeline-graph.scss @@ -1,11 +1,38 @@ +#filter-bar { + @extend .card; + @extend .bg-dark; + + #filter { + width: 100%; + @extend .bg-dark; + @extend .text-light; + @extend .form-control; + @extend .form-control-lg; + @extend .rounded; + @extend .border-0; + + // add search icon + @extend .icon-search-light; + background-position-x: 8px; + padding-left: 2.6rem; + height: unset; + } +} + +@function calculate-graph-height() { + /* 73px of header, 50px of filter, and 50px of footer */ + $graphHeight: calc(100vh - (73px + 50px + 50px)); + @return $graphHeight; +} + #pipeline-graph { position: relative; width: 100%; - height: 1000px; + height: calculate-graph-height(); #graph { width: 100%; - height: 1000px; + height: calculate-graph-height(); display: block; @extend .bg-dark; @extend .text-white; @@ -20,18 +47,55 @@ @extend .small; header { + @extend .d-flex; @extend .card-header; + + #pipelines-key-text { + @extend .flex-grow-1; + @extend .font-weight-bold; + @extend .mr-1; + } + + #pipelines-back { + @extend .icon-arrowleft-light; + @extend .ml-1; + height: 1.5em; + width: 1.5em; + background-size: 1.5em; + vertical-align: text-bottom; + } } ul { @extend .list-group; @extend .list-group-flush; max-height: 500px; - overflow-y: scroll; + overflow-y: auto; li { @extend .list-group-item; @extend .bg-dark; } + li:hover { + @extend .text-light:hover; + } + } + } +} + +.graph-tooltip { + @extend .tooltip; + @extend .show; + font-size: 0.8em; + pointer-events: none; // prevent tooltip from blocking mouse events + + .graph-tooltip-inner { + @extend .tooltip-inner; + @extend .rounded; + @extend .text-left; + + ul { + @extend .mb-0; + padding-left: 1em; } } } diff --git a/app/assets/stylesheets/limber/screen.scss b/app/assets/stylesheets/limber/screen.scss index 60557fbd6..f01531641 100644 --- a/app/assets/stylesheets/limber/screen.scss +++ b/app/assets/stylesheets/limber/screen.scss @@ -123,7 +123,7 @@ header.limber-header { } #app { - margin-bottom: 100px; + margin-bottom: 50px; } #flashes { diff --git a/app/controllers/pipelines_controller.rb b/app/controllers/pipelines_controller.rb index af7b9756a..1f64aa016 100644 --- a/app/controllers/pipelines_controller.rb +++ b/app/controllers/pipelines_controller.rb @@ -36,6 +36,8 @@ def calculate_nodes id: purpose[:name], type: purpose[:asset_type], input: purpose[:input_plate], + stock: purpose[:stock_plate], + cherrypickable_target: purpose[:cherrypickable_target], size: purpose[:size] } } diff --git a/app/javascript/pipeline-graph/filterFunctions.js b/app/javascript/pipeline-graph/filterFunctions.js new file mode 100644 index 000000000..7538548f8 --- /dev/null +++ b/app/javascript/pipeline-graph/filterFunctions.js @@ -0,0 +1,34 @@ +let notResults = undefined +/** + * Searches for nodes and edges in a Cytoscape.js graph that match a given query. + * Nodes are matched if their 'id' attribute contains the query string. + * Edges are matched if their 'pipeline' attribute starts with the query string. + * The function also includes neighboring nodes of matched nodes and connected nodes of matched edges. + * As a side-effect, all non-matching elements are removed from the graph. + * + * @param {cytoscape.Core} cy - The Cytoscape instance (i.e., the graph). + * @param {string} query - The search term. + * @returns {cytoscape.Collection} - A collection of nodes and edges that match the query. + */ +const findResults = (cy, query) => { + if (notResults !== undefined) { + notResults.restore() + } + + const all = cy.$('*') + let results = cy.collection() + + let purposes = cy.$(`node[id @*= "${query}"]`) + purposes = purposes.union(purposes.neighborhood()) + results = results.union(purposes) + + let pipelines = cy.$(`edge[pipeline @^= "${query}"]`) + pipelines = pipelines.union(pipelines.connectedNodes()) + results = results.union(pipelines) + + notResults = cy.remove(all.not(results)) + + return results +} + +export { findResults } diff --git a/app/javascript/pipeline-graph/filterFunctions.spec.js b/app/javascript/pipeline-graph/filterFunctions.spec.js new file mode 100644 index 000000000..3c0a873f1 --- /dev/null +++ b/app/javascript/pipeline-graph/filterFunctions.spec.js @@ -0,0 +1,108 @@ +import { findResults } from './filterFunctions' +import cytoscape from 'cytoscape' + +describe('findResults', () => { + const cy = cytoscape({ + elements: [ + { data: { id: 'plateA1' } }, + { data: { id: 'plateA2' } }, + { data: { id: 'plateB' } }, + { data: { id: 'plateC' } }, + { data: { id: 'edge1', source: 'plateA1', target: 'plateB', pipeline: 'pipeline1' } }, + { data: { id: 'edge2', source: 'plateA2', target: 'plateB', pipeline: 'pipeline2' } }, + { data: { id: 'edge3', source: 'plateB', target: 'plateC', pipeline: 'pipeline3' } }, + ], + }) + + it('should return pipelines and associated purposes that match the query', () => { + const query = 'pipeline1' + const results = findResults(cy, query) + const resultsIds = results.map((ele) => ele.id()) + + expect(resultsIds.length).toEqual(3) + expect(resultsIds).toContain('plateA1') + expect(resultsIds).toContain('plateB') + expect(resultsIds).toContain('edge1') + + // check that non-matching elements have been removed + const all = cy.$('*') + expect(all.length).toEqual(3) + }) + + it('should return purposes and neighboring purposes and pipelines that match the query', () => { + const query = 'plateA' // matches plateA1 and plateA2 + const results = findResults(cy, query) + const resultsIds = results.map((ele) => ele.id()) + + expect(resultsIds.length).toEqual(5) + expect(resultsIds).toContain('plateA1') + expect(resultsIds).toContain('plateA2') + expect(resultsIds).toContain('plateB') + expect(resultsIds).toContain('edge1') + expect(resultsIds).toContain('edge2') + + // check that non-matching elements have been removed + const all = cy.$('*') + expect(all.length).toEqual(5) + }) + + it('should restore non-matching elements from previous search', () => { + const query = 'pipeline1' + const results = findResults(cy, query) + const resultsIds = results.map((ele) => ele.id()) + + expect(resultsIds.length).toEqual(3) + expect(resultsIds).toContain('plateA1') + expect(resultsIds).toContain('plateB') + expect(resultsIds).toContain('edge1') + + // check that non-matching elements have been removed + const all = cy.$('*') + expect(all.length).toEqual(3) + + // search for a different query + const query2 = 'pipeline2' + const results2 = findResults(cy, query2) + const resultsIds2 = results2.map((ele) => ele.id()) + + expect(resultsIds2.length).toEqual(3) + expect(resultsIds2).toContain('plateA2') + expect(resultsIds2).toContain('plateB') + expect(resultsIds2).toContain('edge2') + + // check that non-matching elements have been removed + const all2 = cy.$('*') + expect(all2.length).toEqual(3) + }) + + it('should return empty collection if no elements match the query', () => { + const query = 'no match' + const results = findResults(cy, query) + const resultsIds = results.map((ele) => ele.id()) + + expect(resultsIds.length).toEqual(0) + + // check that non-matching elements have been removed + const all = cy.$('*') + expect(all.length).toEqual(0) + }) + + it('should return all elements if query is empty', () => { + const query = '' + const results = findResults(cy, query) + const resultsIds = results.map((ele) => ele.id()) + + expect(resultsIds.length).toEqual(7) + expect(resultsIds).toContain('plateA1') + expect(resultsIds).toContain('plateA2') + expect(resultsIds).toContain('plateB') + expect(resultsIds).toContain('plateC') + expect(resultsIds).toContain('edge1') + expect(resultsIds).toContain('edge2') + expect(resultsIds).toContain('edge3') + + // check that non-matching elements have been removed + const all = cy.$('*') + expect(all.length).toEqual(7) + }) +}) diff --git a/app/javascript/pipeline-graph/index.js b/app/javascript/pipeline-graph/index.js index 7f1ac07b9..4b740ba5b 100644 --- a/app/javascript/pipeline-graph/index.js +++ b/app/javascript/pipeline-graph/index.js @@ -1,7 +1,13 @@ // Mounts #graph and renders a summary of the limber pipelines source from // pipelines.json +import { findResults } from './filterFunctions' import cytoscape from 'cytoscape' +import elk from 'cytoscape-elk' +import popper from 'cytoscape-popper' + +cytoscape.use(popper) +cytoscape.use(elk) // Distinct colours as used in the rest of limber const colours = [ @@ -103,42 +109,254 @@ const colours = [ '#5B4534', ] +let cy = undefined const pipelineColours = {} +const filterField = document.getElementById('filter') +const pipelinesBackButton = document.getElementById('pipelines-back') +let filterHistory = [''] // start with an empty filter -const pipelineColour = function (node) { - var pipeline = node.data('pipeline') - if (pipelineColours[pipeline] === undefined) { - pipelineColours[pipeline] = colours.shift() || '#666' - } - return pipelineColours[pipeline] +const pipelineColourEdge = function (edge) { + var pipeline = edge.data('pipeline') + return pipelineColours[pipeline] || '#666' +} +const calculatePipelineColours = function (pipelineNames) { + const coloursCopy = [...colours] + pipelineNames.forEach((pipeline) => { + const colour = coloursCopy.shift() + pipelineColours[pipeline] = colour + }) +} + +const renderPipelinesKey = function (pipelineNames) { + const key = document.getElementById('pipelines-key') + key.innerHTML = '' + pipelineNames.forEach((pipeline) => { + const item = document.createElement('li') + const pipelineColour = pipelineColours[pipeline] || '#666' + + item.style.borderLeft = `solid 10px ${pipelineColour}` + item.role = 'button' + item.textContent = pipeline + + // when each pipeline name is hovered over, the corresponding edges are highlighted + item.addEventListener('mouseover', () => { + cy.elements('edge[pipeline = "' + pipeline + '"]').addClass('highlight') + }) + item.addEventListener('mouseout', () => { + cy.elements('edge[pipeline = "' + pipeline + '"]').removeClass('highlight') + }) + + // when each pipeline key name is clicked, filter the graph to show only that pipeline + item.addEventListener('click', () => { + applyFilter(pipeline) + }) + + key.appendChild(item) + }) } // Polygon dimensions represent a series of points defined by alternating x,y co-ordinates // They are bounded by the top left (-1,-1) and the bottom right (1,1) // The polygons here begin with the upper left most position, and proceed round the shape // clockwise -const platePolygon = '-1 -1 0.9 -1 1 -0.9 1 0.9 0.9 1 -0.9 1 -1 0.9' -const tubePolygon = '-1 -1 1 -1 1 -0.9 0.9 -0.9 0.9 0.75 0 1 -0.9 0.75 -0.9 -0.9 -1 -0.9' -const bg384 = - '' + } + + // Create a temporary canvas + const canvas = document.createElement('canvas') + canvas.width = 48 * 3 // set canvas size to match node shape, with extra for zoom + canvas.height = 48 * 3 // set canvas size to match node shape, with extra for zoom + // NOTE: canvas is clipped to shape by cytoscape + + // Get the context of the canvas + const ctx = canvas.getContext('2d') + + if (!isDefaultSize) { + // Write the node size in the center + ctx.font = 'bold 52px sans-serif' + ctx.textAlign = 'center' + ctx.textBaseline = 'middle' + ctx.fillStyle = 'white' + ctx.fillText(size, canvas.width / 2, canvas.height / 2) + } + + // Draw bands to represent the node properties + const bands = { + input: { present: input, colour: 'cyan' }, + stock: { present: stock, colour: 'magenta' }, + cherrypickable_target: { present: cherrypickable_target, colour: 'yellow' }, + } + const bandWidth = 3 * 3 // width of the coloured bands + + let bandOffset = 48 * 3 - bandWidth // start at the right of the canvas + Object.keys(bands).forEach((band) => { + if (bands[band].present) { + ctx.fillStyle = bands[band].colour + ctx.fillRect(bandOffset, 0, bandWidth, canvas.height) + bandOffset -= bandWidth + } + }) + + // Export the canvas to data URI + const dataURI = canvas.toDataURL() + return dataURI +} + +// Dynamically create an icon based on the node's properties +const renderIcon = function (ele) { + const size = ele.data('size') + const input = ele.data('input') + const stock = ele.data('stock') + const cherrypickable_target = ele.data('cherrypickable_target') + return renderCanvas(size, input, stock, cherrypickable_target) +} + +const generateTooltipContent = function (ele) { + // pipeline properties + const pipelineName = ele.data('pipeline') + + // purpose node properties + const purposeName = ele.data('id') + const purposeType = ele.data('type') + const size = ele.data('size') + const isInput = ele.data('input') + const isStock = ele.data('stock') + const isCherrypickableTarget = ele.data('cherrypickable_target') + + let content = '' + // if element is an edge + if (ele.isEdge()) { + content = pipelineName + } else { + content = `${purposeName} [${purposeType}]` + + const properties = { + nonStandardSize: { present: size !== null && size != 96, label: `Size: ${size}` }, + input: { present: isInput, label: 'Input' }, + stock: { present: isStock, label: 'Stock' }, + cherrypickableTarget: { present: isCherrypickableTarget, label: 'Cherrypickable Target' }, + } + + // if any properties are present + if (Object.values(properties).some((property) => property.present)) { + content += '
    ' + Object.keys(properties).forEach((property) => { + if (properties[property].present) { + content += `
  • ${properties[property].label}
  • ` + } + }) + content += '
' + } + } + + return content +} + +const applyMouseEvents = function () { + cy.elements().unbind('mouseover') + cy.elements().bind('mouseover', (event) => { + // Highlight when mouse enters element + event.target.addClass('highlight') + + // Add popper when mouse enters element + event.target.popperRefObj = event.target.popper({ + content: () => { + const content = generateTooltipContent(event.target) + + document.body.insertAdjacentHTML( + 'beforeend', + `
+
+ ${content} +
+
` + ) + return document.querySelector('.graph-tooltip') + }, + }) + }) + + cy.elements().unbind('mouseout') + cy.elements().bind('mouseout', (event) => { + // Remove highlight when mouse leaves element + event.target.removeClass('highlight') + + // Remove popper when mouse leaves element + if (event.target.popper) { + event.target.popperRefObj.state.elements.popper.remove() + event.target.popperRefObj.destroy() + } + }) + + // when an edge is clicked, filter the graph to show only that pipeline + cy.on('click', 'edge', (event) => { + const pipeline = event.target.data('pipeline') + applyFilter(pipeline) + }) +} + +// for other layout options see https://js.cytoscape.org/#demos +// for elk options see https://eclipse.dev/elk/reference/algorithms/org-eclipse-elk-layered.html +const layoutOptions = { + name: 'elk', + transform: function (node, pos) { + // A function that applies a transform to the final node position + // scale x coordinates to prevent labels overlapping + pos.x *= 2.25 + pos.y *= 1.35 + return pos + }, + elk: { + algorithm: 'layered', + 'elk.direction': 'DOWN', + }, +} const renderPipelines = function (data) { + const pipelines = data.pipelines.map((pipeline) => pipeline.name).sort() + calculatePipelineColours(pipelines) + renderPipelinesKey(pipelines) + const container = document.getElementById('graph') - const key = document.getElementById('pipelines-key') - cytoscape({ + cy = cytoscape({ container, // container to render in elements: data.elements, + autoungrabify: true, // don't allow nodes to be dragged around + + // node dimensions: + // - type: plate or tube + // - size: number of wells + // - input: true if the node is an input to the pipeline + // TODO: add legend for the above + style: [ // the stylesheet for the graph { selector: 'node', style: { 'background-color': '#666', + 'background-image': renderIcon, + 'background-height': '100%', // set canvas to fit node + 'background-width': '100%', // set canvas to fit node label: 'data(id)', color: 'white', + width: 48, + height: 48, + 'text-halign': 'right', + 'text-valign': 'center', + 'text-wrap': 'wrap', + 'text-max-width': '90', + 'text-margin-x': '5', }, }, { @@ -146,14 +364,6 @@ const renderPipelines = function (data) { style: { shape: 'polygon', 'shape-polygon-points': platePolygon, - width: 127.76 / 2, - height: 85.47 / 2, - }, - }, - { - selector: '[size = 384]', - style: { - 'background-image': bg384, }, }, { @@ -161,14 +371,6 @@ const renderPipelines = function (data) { style: { shape: 'polygon', 'shape-polygon-points': tubePolygon, - width: 27 / 2, - height: 80 / 2, - }, - }, - { - selector: '[?input]', - style: { - 'background-color': '#bbb', }, }, { @@ -176,37 +378,93 @@ const renderPipelines = function (data) { style: { width: 3, 'curve-style': 'bezier', - 'line-color': pipelineColour, - 'target-arrow-color': pipelineColour, + 'control-point-step-size': 15, + 'line-color': pipelineColourEdge, + 'target-arrow-color': pipelineColourEdge, 'target-arrow-shape': 'triangle', - 'arrow-scale': 3, + 'arrow-scale': 2, + }, + }, + { + selector: 'edge.highlight', + style: { + 'underlay-opacity': 0.3, }, }, ], - // for other layout options see http://js.cytoscape.org/#layouts - layout: { - name: 'cose', - nodeDimensionsIncludeLabels: true, - idealEdgeLength: 90, - nodeRepulsion: 100000, - gravity: 0.01, - animate: false, - }, + layout: layoutOptions, minZoom: 0.2, - maxZoom: 3, + maxZoom: 3, // referenced in renderIcon above }) - data.pipelines.forEach((pipeline) => { - const item = document.createElement('li') - item.style = 'border-left: solid 10px ' + pipelineColours[pipeline.name] + ';' - item.innerHTML = pipeline.name - key.appendChild(item) - }) + applyMouseEvents() } // Fetch the result of pipelines.json and then render the graph. +// pipelines.json is generated by the pipelines controller fetch('pipelines.json').then((response) => { - response.json().then(renderPipelines) + response.json().then((data) => { + // Get filter from url + const url = new URL(window.location.href) + const filter = url.searchParams.get('filter') + filterField.value = filter // set before rendering the graph for the users benefit + + // Render the graph + renderPipelines(data) + + // Apply filter if present + if (filter) { + applyFilter(filter) + } + }) +}) + +const applyFilter = function (filter) { + // set value in filter field + filterField.value = filter + + // set url to reflect filter + const url = new URL(window.location.href) + url.searchParams.set('filter', filter) + window.history.pushState({}, '', url) + + // set page title to reflect filter + document.title = `Limber - Pipelines - ${filter}` + + // show or hide back button + if (filterHistory.length > 0) { + pipelinesBackButton.classList.remove('invisible') + } else { + pipelinesBackButton.classList.add('invisible') + } + + // add filter to (internal - not browser) history + if (filterHistory[filterHistory.length - 1] !== filter) { + // don't add duplicate filters + filterHistory.push(filter) + } + + // apply filter to graph + const results = findResults(cy, filter) + + const pipelineNames = [...new Set(results.edges().map((edge) => edge.data('pipeline')))].sort() + calculatePipelineColours(pipelineNames) + renderPipelinesKey(pipelineNames) + + results.layout(layoutOptions).run() +} + +filterField.addEventListener('change', (event) => { + const query = event.target.value + applyFilter(query) +}) + +pipelinesBackButton.addEventListener('click', () => { + // remove and apply previous filter + filterHistory.pop() // remove current filter + const previousFilter = filterHistory.pop() + + applyFilter(previousFilter) }) diff --git a/app/models/labware_creators/cardinal_pools_plate.rb b/app/models/labware_creators/cardinal_pools_plate.rb index d34e7c0cc..ece69eeb7 100644 --- a/app/models/labware_creators/cardinal_pools_plate.rb +++ b/app/models/labware_creators/cardinal_pools_plate.rb @@ -126,7 +126,7 @@ def build_pools current_pool = 0 # wells_grouped_by_collected_by = {0=>['w1', 'w4'], 1=>['w6', 'w2'], 2=>['w9', 'w23']} - wells_grouped_by_collected_by.each do |_collected_by, wells| + wells_grouped_by_collected_by.each_value do |wells| # Loop through the wells for that collected_by wells.each do |well| # Create pool if it doesnt already exist diff --git a/app/models/labware_creators/plate_split_to_tube_racks.rb b/app/models/labware_creators/plate_split_to_tube_racks.rb index 8369cee2d..d073b4ef7 100644 --- a/app/models/labware_creators/plate_split_to_tube_racks.rb +++ b/app/models/labware_creators/plate_split_to_tube_racks.rb @@ -375,7 +375,7 @@ def find_parent_wells_for_sequencing unique_sample_uuids = [] parent_wells_for_seq = [] - well_filter.filtered.each do |well, _ignore| + well_filter.filtered.each do |(well, _ignore)| sample_uuid = well.aliquots.first.sample.uuid next if sample_uuid.in?(unique_sample_uuids) diff --git a/app/models/labware_creators/quadrant_stamp_base.rb b/app/models/labware_creators/quadrant_stamp_base.rb index 27585f29b..bb1ddbed6 100644 --- a/app/models/labware_creators/quadrant_stamp_base.rb +++ b/app/models/labware_creators/quadrant_stamp_base.rb @@ -59,7 +59,7 @@ def stock_barcodes_by_quadrant source_plate = Sequencescape::Api::V2::Plate.find_by(uuid: uuid) stock_barcode = source_plate&.stock_plate&.barcode&.human - quadrants["stock_barcode_q#{index}".to_sym] = stock_barcode unless stock_barcode.nil? + quadrants[:"stock_barcode_q#{index}"] = stock_barcode unless stock_barcode.nil? end quadrants end diff --git a/app/models/labware_creators/stamped_plate_adding_randomised_controls.rb b/app/models/labware_creators/stamped_plate_adding_randomised_controls.rb index 120785349..720984c0d 100644 --- a/app/models/labware_creators/stamped_plate_adding_randomised_controls.rb +++ b/app/models/labware_creators/stamped_plate_adding_randomised_controls.rb @@ -284,7 +284,7 @@ def suitable_request_for_well(parent_well_v2) request.request_type.key == purpose_config.fetch(:work_completion_request_type) && request.state == 'pending' end - reqs&.sort_by(&:id)&.last + reqs&.max_by(&:id) end # find and close request of type specified by config in the parent well diff --git a/app/models/limber/tag_layout_template.rb b/app/models/limber/tag_layout_template.rb index 11c2cc57e..8e5ed377d 100644 --- a/app/models/limber/tag_layout_template.rb +++ b/app/models/limber/tag_layout_template.rb @@ -22,7 +22,7 @@ def group_wells(plate) prior_pool = nil callback = lambda do |row_column| - prior_pool = pool = (well_to_pool[row_column] || prior_pool) # or next + prior_pool = pool = well_to_pool[row_column] || prior_pool # or next well_empty = well_to_pool[row_column].nil? well = pool.nil? ? nil : row_column [well, pool, well_empty] # Triplet: [ A1, pool_id, well_empty ] diff --git a/app/models/utility/concentration_binning_calculator.rb b/app/models/utility/concentration_binning_calculator.rb index 42ed2a6f5..8ad51ae23 100644 --- a/app/models/utility/concentration_binning_calculator.rb +++ b/app/models/utility/concentration_binning_calculator.rb @@ -90,7 +90,7 @@ def build_transfers_hash(bins, number_of_rows, compression_reqd) # rubocop:todo } # work out what the next row and column will be - finished = ((bin_index_within_bins == bins.size - 1) && (well_index_within_bin == bin.size - 1)) + finished = (bin_index_within_bins == bins.size - 1) && (well_index_within_bin == bin.size - 1) binner.next_well_location(well_index_within_bin, bin.size) unless finished end end diff --git a/app/models/utility/normalised_binning_calculator.rb b/app/models/utility/normalised_binning_calculator.rb index 8bb22b30d..8414541a1 100644 --- a/app/models/utility/normalised_binning_calculator.rb +++ b/app/models/utility/normalised_binning_calculator.rb @@ -79,7 +79,7 @@ def build_transfers_hash(bins, number_of_rows, compression_reqd) # rubocop:todo } # work out what the next row and column will be - finished = ((bin_index_within_bins == bins.size - 1) && (well_index_within_bin == bin.size - 1)) + finished = (bin_index_within_bins == bins.size - 1) && (well_index_within_bin == bin.size - 1) binner.next_well_location(well_index_within_bin, bin.size) unless finished end end diff --git a/app/models/utility/pcr_cycles_binning_calculator.rb b/app/models/utility/pcr_cycles_binning_calculator.rb index 5fb52b741..60dc0f490 100644 --- a/app/models/utility/pcr_cycles_binning_calculator.rb +++ b/app/models/utility/pcr_cycles_binning_calculator.rb @@ -43,7 +43,7 @@ def compute_presenter_bin_details def calculate_bins bins = [] - @well_details.each do |_well_locn, details| + @well_details.each_value do |details| pcr_cycles = details['pcr_cycles'] bins << pcr_cycles unless bins.include? pcr_cycles end @@ -90,7 +90,7 @@ def build_transfers_well(binner, transfers_hash, well) # work out what the next row and column will be def binner_next_well(binner, bins, bin, bin_index_within_bins, well_index_within_bin) - finished = ((bin_index_within_bins == bins.size - 1) && (well_index_within_bin == bin.size - 1)) + finished = (bin_index_within_bins == bins.size - 1) && (well_index_within_bin == bin.size - 1) binner.next_well_location(well_index_within_bin, bin.size) unless finished end end diff --git a/app/views/pipelines/index.html.erb b/app/views/pipelines/index.html.erb index 27e929762..4d8f82e90 100644 --- a/app/views/pipelines/index.html.erb +++ b/app/views/pipelines/index.html.erb @@ -1,10 +1,15 @@ +
+ +
-
Key
+
+ Pipelines Key + +
<%= javascript_pack_tag 'pipeline-graph' %> - diff --git a/config/application.rb b/config/application.rb index 13a84c719..2bae83bac 100644 --- a/config/application.rb +++ b/config/application.rb @@ -27,6 +27,7 @@ module Limber class Application < Rails::Application # rubocop:todo Style/Documentation config.load_defaults 6.0 + config.active_support.cache_format_version = 7.0 # Settings in config/environments/* take precedence over those specified here. # Application configuration should go into files in config/initializers diff --git a/lib/config_loader/base.rb b/lib/config_loader/base.rb index fc0f6603f..18664f237 100644 --- a/lib/config_loader/base.rb +++ b/lib/config_loader/base.rb @@ -54,7 +54,7 @@ def default_path end def in_list?(list, file) - (list.nil? || list.include?(file.basename(EXTENSION).to_s)) + list.nil? || list.include?(file.basename(EXTENSION).to_s) end def work_in_progress?(filename) diff --git a/lib/purpose_config.rb b/lib/purpose_config.rb index 1b3aea1a4..a67954d26 100644 --- a/lib/purpose_config.rb +++ b/lib/purpose_config.rb @@ -35,7 +35,7 @@ def initialize(name, options, store, api, submission_templates, label_templates) @api = api @submission_templates = submission_templates @label_templates = label_templates - @template_name = (@options.delete(:label_template) || '') + @template_name = @options.delete(:label_template) || '' end def config diff --git a/package.json b/package.json index 005c7e4df..5b47549d5 100644 --- a/package.json +++ b/package.json @@ -6,6 +6,8 @@ "babel-loader": "^8.2.5", "bootstrap-vue": "^2.23.1", "cytoscape": "^3.28.1", + "cytoscape-elk": "^2.2.0", + "cytoscape-popper": "^2.0.0", "devour-client": "^2.1.2", "flush-promises": "^1.0.0", "pluralize": "^7.0.0", @@ -13,6 +15,7 @@ "vue-loader": "^15.11.1", "vue-style-loader": "^4.1.3", "vue-template-compiler": "^2.6.11", + "web-worker": "^1.3.0", "webpack": "^4.46.0", "webpack-cli": "^3.3.12" }, diff --git a/spec/support/factory_bot_extensions.rb b/spec/support/factory_bot_extensions.rb index 4acbf617c..54c26c611 100644 --- a/spec/support/factory_bot_extensions.rb +++ b/spec/support/factory_bot_extensions.rb @@ -23,15 +23,16 @@ class DefinitionProxy def with_has_many_associations(*names, actions: ['read']) transient do names.each do |association| - send("#{association}_count") { 0 } - send("#{association}_actions") { actions } + send(:"#{association}_count") { 0 } + send(:"#{association}_actions") { actions } end end names.each do |association| send(association) do {}.tap do |h| - h['size'] = send("#{association}_count") if send("#{association}_actions").include?('read') - h['actions'] = send("#{association}_actions").index_with { |_action_name| "#{resource_url}/#{association}" } + h['size'] = send(:"#{association}_count") if send(:"#{association}_actions").include?('read') + h['actions'] = + send(:"#{association}_actions").index_with { |_action_name| "#{resource_url}/#{association}" } end end end @@ -41,16 +42,16 @@ def with_has_many_associations(*names, actions: ['read']) def with_belongs_to_associations(*names, actions: ['read']) transient do names.each do |association| - send("#{association}_uuid") { "#{association}-uuid" } - send("#{association}_actions") { actions } + send(:"#{association}_uuid") { "#{association}-uuid" } + send(:"#{association}_actions") { actions } end end names.each do |association| send(association) do { 'actions' => - send("#{association}_actions").index_with { |_action_name| api_root + send("#{association}_uuid") }, - 'uuid' => send("#{association}_uuid") + send(:"#{association}_actions").index_with { |_action_name| api_root + send(:"#{association}_uuid") }, + 'uuid' => send(:"#{association}_uuid") } end end diff --git a/yarn.lock b/yarn.lock index 8e6655a2e..836511c31 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2358,6 +2358,11 @@ consola "^2.15.0" node-fetch "^2.6.1" +"@popperjs/core@^2.0.0": + version "2.11.8" + resolved "https://registry.yarnpkg.com/@popperjs/core/-/core-2.11.8.tgz#6b79032e760a0899cd4204710beede972a3a185f" + integrity sha512-P1st0aksCrn9sGZhp8GMYwBnQsbvAWsZAX44oXNNvLHGqAOcoVxmjZiohstwQ7SqKnbR47akdNi+uleWD8+g6A== + "@prettier/plugin-ruby@^2.1.0": version "2.1.0" resolved "https://registry.yarnpkg.com/@prettier/plugin-ruby/-/plugin-ruby-2.1.0.tgz#2df48c1c004fc9dec18a72f441d43d54137b25a6" @@ -4655,6 +4660,20 @@ cyclist@^1.0.1: resolved "https://registry.yarnpkg.com/cyclist/-/cyclist-1.0.1.tgz#596e9698fd0c80e12038c2b82d6eb1b35b6224d9" integrity sha1-WW6WmP0MgOEgOMK4LW6xs1tiJNk= +cytoscape-elk@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/cytoscape-elk/-/cytoscape-elk-2.2.0.tgz#330364b4c799f59179904a775c864b725018e7d4" + integrity sha512-EqXBVRcWeah/oBOifAmne0ImmIKntBVEQh2XCJXY++BgCufehZglRclrJ1DWm5Qm/NDBO/wEDijjgd50xJXw0A== + dependencies: + elkjs "^0.8.1" + +cytoscape-popper@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/cytoscape-popper/-/cytoscape-popper-2.0.0.tgz#d93917695a9b8af3dbda1d8ee433618ac4d4e359" + integrity sha512-b7WSOn8qXHWtdIXFNmrgc8qkaOs16tMY0EwtRXlxzvn8X+al6TAFrUwZoYATkYSlotfd/36ZMoeKMEoUck6feA== + dependencies: + "@popperjs/core" "^2.0.0" + cytoscape@^3.28.1: version "3.28.1" resolved "https://registry.yarnpkg.com/cytoscape/-/cytoscape-3.28.1.tgz#f32c3e009bdf32d47845a16a4cd2be2bbc01baf7" @@ -4962,6 +4981,11 @@ electron-to-chromium@^1.4.526: resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.530.tgz#c31a44346739bb34acb1a4026a07c3b9eeeb326c" integrity sha512-rsJ9O8SCI4etS8TBsXuRfHa2eZReJhnGf5MHZd3Vo05PukWHKXhk3VQGbHHnDLa8nZz9woPCpLCMQpLGgkGNRA== +elkjs@^0.8.1: + version "0.8.2" + resolved "https://registry.yarnpkg.com/elkjs/-/elkjs-0.8.2.tgz#c37763c5a3e24e042e318455e0147c912a7c248e" + integrity sha512-L6uRgvZTH+4OF5NE/MBbzQx/WYpru1xCBE9respNj6qznEewGUIfhzmm7horWWxbNO2M0WckQypGctR8lH79xQ== + elliptic@^6.0.0, elliptic@^6.5.4: version "6.5.4" resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.4.tgz#da37cebd31e79a1367e941b592ed1fbebd58abbb" @@ -5597,9 +5621,9 @@ flush-write-stream@^1.0.0: readable-stream "^2.3.6" follow-redirects@^1.0.0, follow-redirects@^1.14.0: - version "1.14.8" - resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.14.8.tgz#016996fb9a11a100566398b1c6839337d7bfa8fc" - integrity sha512-1x0S9UVJHsQprFcEC/qnNzBLcIxsjAV905f/UkQxbclCsoTWlacCNOpQa/anodLl2uaEKFhfWOvM2Qg77+15zA== + version "1.15.4" + resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.4.tgz#cdc7d308bf6493126b17ea2191ea0ccf3e535adf" + integrity sha512-Cr4D/5wlrb0z9dgERpUL3LrmPKVDsETIJhaCMeDfuFYcqa5bldGV6wBsAN6X/vxlXQtFBMrXdXxdL8CbDTGniw== for-in@^1.0.2: version "1.0.2" @@ -10866,6 +10890,11 @@ wbuf@^1.1.0, wbuf@^1.7.3: dependencies: minimalistic-assert "^1.0.0" +web-worker@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/web-worker/-/web-worker-1.3.0.tgz#e5f2df5c7fe356755a5fb8f8410d4312627e6776" + integrity sha512-BSR9wyRsy/KOValMgd5kMyr3JzpdeoR9KVId8u5GVlTTAtNChlsE4yTxeY7zMdNSyOmoKBv8NH2qeRY9Tg+IaA== + webidl-conversions@^3.0.0: version "3.0.1" resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871"