mirror of
				https://github.com/scinote-eln/scinote-web.git
				synced 2025-10-26 22:16:28 +08:00 
			
		
		
		
	Add report results to global search page [SCI-10479]
This commit is contained in:
		
							parent
							
								
									6c5efb604f
								
							
						
					
					
						commit
						d76fbf1eba
					
				
					 7 changed files with 129 additions and 3 deletions
				
			
		|  | @ -42,6 +42,21 @@ class SearchController < ApplicationController | ||||||
|                    next_page: results.try(:next_page) |                    next_page: results.try(:next_page) | ||||||
|                  } |                  } | ||||||
|           return |           return | ||||||
|  |         when 'reports' | ||||||
|  |           @report_search_count = fetch_cached_count Report | ||||||
|  |           search_reports | ||||||
|  |           results = if params[:preview] == 'true' | ||||||
|  |                       @report_results.limit(Constants::GLOBAL_SEARCH_PREVIEW_LIMIT) | ||||||
|  |                     else | ||||||
|  |                       @report_results.page(params[:page]).per(Constants::SEARCH_LIMIT) | ||||||
|  |                     end | ||||||
|  |           render json: results.includes(:team, :project, :user), | ||||||
|  |                  each_serializer: GlobalSearch::ReportSerializer, | ||||||
|  |                  meta: { | ||||||
|  |                    total: @search_count, | ||||||
|  |                    next_page: results.try(:next_page) | ||||||
|  |                  } | ||||||
|  |           return | ||||||
|         when 'experiments' |         when 'experiments' | ||||||
|           @experiment_search_count = fetch_cached_count Experiment |           @experiment_search_count = fetch_cached_count Experiment | ||||||
|           search_experiments |           search_experiments | ||||||
|  | @ -348,7 +363,7 @@ class SearchController < ApplicationController | ||||||
|   end |   end | ||||||
| 
 | 
 | ||||||
|   def search_reports |   def search_reports | ||||||
|     @report_results = [] |     @report_results = Report.none | ||||||
|     @report_results = search_by_name(Report) if @report_search_count.positive? |     @report_results = search_by_name(Report) if @report_search_count.positive? | ||||||
|     @search_count = @report_search_count |     @search_count = @report_search_count | ||||||
|   end |   end | ||||||
|  |  | ||||||
|  | @ -1,14 +1,66 @@ | ||||||
| <template> | <template> | ||||||
|   <div class="bg-white rounded p-4 mb-4"> |   <div v-if="total" class="bg-white rounded p-4 mb-4"> | ||||||
|     <h2 class="flex items-center gap-2 mt-0 mb-4"> |     <h2 class="flex items-center gap-2 mt-0 mb-4"> | ||||||
|       <i class="sn-icon sn-icon-reports"></i> |       <i class="sn-icon sn-icon-reports"></i> | ||||||
|       {{ i18n.t('search.index.reports') }} |       {{ i18n.t('search.index.reports') }} | ||||||
|  |       [{{ total }}] | ||||||
|     </h2> |     </h2> | ||||||
|  |     <div> | ||||||
|  |       <div class="grid grid-cols-[auto_110px_auto_auto_auto_auto_auto] items-center"> | ||||||
|  |         <template v-for="row in preparedResults" :key="row.id"> | ||||||
|  |           <a target="_blank" :href="row.attributes.url" class="h-full py-2 px-4 overflow-hidden font-bold border-0 border-b border-solid border-sn-light-grey"> | ||||||
|  |             <StringWithEllipsis class="w-full" :text="row.attributes.name"></StringWithEllipsis> | ||||||
|  |           </a> | ||||||
|  |           <div class="h-full py-2 px-4 flex items-center gap-1 text-xs border-0 border-b border-solid border-sn-light-grey"> | ||||||
|  |             <b class="shrink-0">{{ i18n.t('search.index.id') }}:</b> | ||||||
|  |             <span class="shrink-0">{{ row.attributes.code }}</span> | ||||||
|  |           </div> | ||||||
|  |           <div class="h-full py-2 px-4 flex items-center gap-1 text-xs border-0 border-b border-solid border-sn-light-grey"> | ||||||
|  |             <b class="shrink-0">{{ i18n.t('search.index.created_at') }}:</b> | ||||||
|  |             <span class="shrink-0">{{ row.attributes.created_at }}</span> | ||||||
|  |           </div> | ||||||
|  |           <div class="h-full py-2 px-4 flex items-center gap-1 text-xs border-0 border-b border-solid border-sn-light-grey"> | ||||||
|  |             <b class="shrink-0">{{ i18n.t('search.index.updated_at') }}:</b> | ||||||
|  |             <span class="shrink-0">{{ row.attributes.updated_at }}</span> | ||||||
|  |           </div> | ||||||
|  |           <div class="h-full py-2 px-4 grid grid-cols-[auto_1fr] items-center gap-1 text-xs border-0 border-b border-solid border-sn-light-grey"> | ||||||
|  |             <b class="shrink-0">{{ i18n.t('search.index.created_by') }}:</b> | ||||||
|  |             <div class="truncate"> | ||||||
|  |               <img :src="row.attributes.created_by.avatar_url" class="w-5 h-5 border border-sn-super-light-grey rounded-full mx-1" /> | ||||||
|  |               <span class="truncate">{{ row.attributes.created_by.name }}</span> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |           <div class="h-full py-2 px-4 grid grid-cols-[auto_1fr] items-center gap-1 text-xs border-0 border-b border-solid border-sn-light-grey"> | ||||||
|  |             <b class="shrink-0">{{ i18n.t('search.index.team') }}:</b> | ||||||
|  |             <a target="_blank" :href="row.attributes.team.url" class="shrink-0 overflow-hidden"> | ||||||
|  |               <StringWithEllipsis class="w-full" :text="row.attributes.team.name"></StringWithEllipsis> | ||||||
|  |             </a> | ||||||
|  |           </div> | ||||||
|  |           <div class="h-full py-2 px-4 grid grid-cols-[auto_1fr] items-center gap-1 text-xs border-0 border-b border-solid border-sn-light-grey"> | ||||||
|  |             <b class="shrink-0">{{ i18n.t('search.index.project') }}:</b> | ||||||
|  |             <a target="_blank" :href="row.attributes.project.url" class="shrink-0 overflow-hidden"> | ||||||
|  |               <StringWithEllipsis class="w-full" :text="row.attributes.project.name"></StringWithEllipsis> | ||||||
|  |             </a> | ||||||
|  |           </div> | ||||||
|  |         </template> | ||||||
|  |       </div> | ||||||
|  |       <div v-if="viewAll" class="mt-4"> | ||||||
|  |         <button class="btn btn-light" @click="$emit('selectGroup', 'ReportsComponent')">View all</button> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|   </div> |   </div> | ||||||
| </template> | </template> | ||||||
| 
 | 
 | ||||||
| <script> | <script> | ||||||
|  | import searchMixin from './search_mixin'; | ||||||
|  | 
 | ||||||
| export default { | export default { | ||||||
|   name: 'ReportsComponent' |   name: 'ReportsComponent', | ||||||
|  |   mixins: [searchMixin], | ||||||
|  |   data() { | ||||||
|  |     return { | ||||||
|  |       group: 'reports' | ||||||
|  |     }; | ||||||
|  |   } | ||||||
| }; | }; | ||||||
| </script> | </script> | ||||||
|  |  | ||||||
|  | @ -48,6 +48,9 @@ import UpdateReportModal from './modals/update.vue'; | ||||||
| 
 | 
 | ||||||
| export default { | export default { | ||||||
|   name: 'ReportsTable', |   name: 'ReportsTable', | ||||||
|  |   provide() { | ||||||
|  |     return { initSearchValue: this.initSearchValue }; | ||||||
|  |   }, | ||||||
|   components: { |   components: { | ||||||
|     DataTable, |     DataTable, | ||||||
|     DocxRenderer, |     DocxRenderer, | ||||||
|  | @ -61,6 +64,10 @@ export default { | ||||||
|       type: String, |       type: String, | ||||||
|       required: true |       required: true | ||||||
|     }, |     }, | ||||||
|  |     initSearchValue: { | ||||||
|  |       type: String, | ||||||
|  |       default: '' | ||||||
|  |     }, | ||||||
|     actionsUrl: { |     actionsUrl: { | ||||||
|       type: String, |       type: String, | ||||||
|       required: true |       required: true | ||||||
|  |  | ||||||
|  | @ -120,6 +120,7 @@ import RowMenuRenderer from './row_menu_renderer.vue'; | ||||||
| 
 | 
 | ||||||
| export default { | export default { | ||||||
|   name: 'App', |   name: 'App', | ||||||
|  |   inject: ['initSearchValue'], | ||||||
|   props: { |   props: { | ||||||
|     withCheckboxes: { |     withCheckboxes: { | ||||||
|       type: Boolean, |       type: Boolean, | ||||||
|  | @ -334,6 +335,9 @@ export default { | ||||||
|   mounted() { |   mounted() { | ||||||
|     this.navigatorCollapsed = document.querySelector('.sci--layout').getAttribute('data-navigator-collapsed') === 'true'; |     this.navigatorCollapsed = document.querySelector('.sci--layout').getAttribute('data-navigator-collapsed') === 'true'; | ||||||
|     this.setGridColsClass(); |     this.setGridColsClass(); | ||||||
|  |     if (this.initSearchValue) { | ||||||
|  |       this.searchValue = this.initSearchValue; | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     window.addEventListener('resize', this.resize); |     window.addEventListener('resize', this.resize); | ||||||
|   }, |   }, | ||||||
|  |  | ||||||
|  | @ -174,6 +174,11 @@ export default { | ||||||
|       type: Object |       type: Object | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|  |   mounted() { | ||||||
|  |     if (this.searchValue.length > 0) { | ||||||
|  |       this.openSearch(); | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|   components: { |   components: { | ||||||
|     MenuDropdown, |     MenuDropdown, | ||||||
|     FilterDropdown, |     FilterDropdown, | ||||||
|  |  | ||||||
							
								
								
									
										42
									
								
								app/serializers/global_search/report_serializer.rb
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								app/serializers/global_search/report_serializer.rb
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,42 @@ | ||||||
|  | # frozen_string_literal: true | ||||||
|  | 
 | ||||||
|  | module GlobalSearch | ||||||
|  |   class ReportSerializer < ActiveModel::Serializer | ||||||
|  |     include Rails.application.routes.url_helpers | ||||||
|  | 
 | ||||||
|  |     attributes :id, :name, :code, :created_at, :updated_at, :created_by, :team, :project, :url | ||||||
|  | 
 | ||||||
|  |     def team | ||||||
|  |       { | ||||||
|  |         name: object.team.name, | ||||||
|  |         url: projects_path(team: object.team) | ||||||
|  |       } | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     def project | ||||||
|  |       { | ||||||
|  |         name: object.project.name, | ||||||
|  |         url: project_experiments_path(project_id: object.project.id) | ||||||
|  |       } | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     def created_by | ||||||
|  |       { | ||||||
|  |         name: object.created_by.name, | ||||||
|  |         avatar_url: avatar_path(object.created_by, :icon_small) | ||||||
|  |       } | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     def created_at | ||||||
|  |       I18n.l(object.created_at, format: :full_date) | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     def updated_at | ||||||
|  |       I18n.l(object.updated_at, format: :full_date) | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     def url | ||||||
|  |       reports_path(search: object.code) | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | @ -17,6 +17,7 @@ | ||||||
|       available-columns-url="<%= available_asset_type_columns_path %>" |       available-columns-url="<%= available_asset_type_columns_path %>" | ||||||
|       available-rows-url="<%= available_rows_repositories_path %>" |       available-rows-url="<%= available_rows_repositories_path %>" | ||||||
|       data-source="<%= reports_path(format: :json) %>" |       data-source="<%= reports_path(format: :json) %>" | ||||||
|  |       init-search-value="<%= params[:search] %>" | ||||||
|       create-url="<%= new_report_path if can_create_reports?(current_team) %>" |       create-url="<%= new_report_path if can_create_reports?(current_team) %>" | ||||||
|     /> |     /> | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		
		Reference in a new issue