Schedule divisions and workshops table
This commit is contained in:
		
							parent
							
								
									c1f9b5db82
								
							
						
					
					
						commit
						ad3136a5cb
					
				
							
								
								
									
										1
									
								
								app/assets/images/admin/workshops.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								app/assets/images/admin/workshops.svg
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1 @@
 | 
			
		||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 26.25"><path d="M0 21h24V0H0v21zM19 1h4v3h-4V1zm0 4h4v3h-4V5zm0 4h4v3h-4V9zm0 4h4v3h-4v-3zm0 4h4v3h-4v-3zM14 1h4v3h-4V1zm0 4h4v3h-4V5zm0 4h4v3h-4V9zm0 4h4v3h-4v-3zm0 4h4v3h-4v-3zM9 1h4v3H9V1zm0 4h4v3H9V5zm0 4h4v3H9V9zm0 4h4v3H9v-3zm0 4h4v3H9v-3zM1 1h7v3H1V1zm0 4h7v3H1V5zm0 4h7v3H1V9zm0 4h7v3H1v-3zm0 4h7v3H1v-3z"/></svg>
 | 
			
		||||
| 
		 After Width: | Height: | Size: 377 B  | 
@ -1,114 +1,114 @@
 | 
			
		||||
(function() {
 | 
			
		||||
	function closeWorkshopSelector() {
 | 
			
		||||
		document.getElementById('workshop-selector').classList.remove('open');
 | 
			
		||||
		document.body.classList.remove('modal-open');
 | 
			
		||||
	}
 | 
			
		||||
	document.getElementById('workshop-selector').addEventListener('click', function(event) {
 | 
			
		||||
		if (event.target.id == 'workshop-selector') {
 | 
			
		||||
			closeWorkshopSelector();
 | 
			
		||||
		}
 | 
			
		||||
	});
 | 
			
		||||
	function _post(form, params, f) {
 | 
			
		||||
		var request = new XMLHttpRequest();
 | 
			
		||||
		request.onreadystatechange = function() {
 | 
			
		||||
			if (request.readyState == 4) {
 | 
			
		||||
				if (request.status == 200) {
 | 
			
		||||
					f(request.responseText);
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		request.open('POST', form.getAttribute('action'), true);
 | 
			
		||||
		request.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded; charset=UTF-8');
 | 
			
		||||
		params['authenticity_token'] = form.querySelector('[name="authenticity_token"]').value;
 | 
			
		||||
		var data = [];
 | 
			
		||||
		for (var key in params) {
 | 
			
		||||
			data.push(key + '=' + params[key]);
 | 
			
		||||
		}
 | 
			
		||||
		request.send(data.join('&'));
 | 
			
		||||
	}
 | 
			
		||||
	function selectorMatches(el, selector) {
 | 
			
		||||
		var p = Element.prototype;
 | 
			
		||||
		var f = p.matches || p.webkitMatchesSelector || p.mozMatchesSelector || p.msMatchesSelector || function(s) {
 | 
			
		||||
			return [].indexOf.call(document.querySelectorAll(s), this) !== -1;
 | 
			
		||||
		};
 | 
			
		||||
		return f.call(el, selector);
 | 
			
		||||
	}
 | 
			
		||||
	function updateSchedule(html) {
 | 
			
		||||
		var schedule = document.getElementById('schedule-preview');
 | 
			
		||||
		var s = document.createElement('div');
 | 
			
		||||
		s.innerHTML = html;
 | 
			
		||||
		schedule.innerHTML = s.children[0].innerHTML;
 | 
			
		||||
		schedule.classList.remove('requesting');
 | 
			
		||||
	}
 | 
			
		||||
  function closeWorkshopSelector() {
 | 
			
		||||
    document.getElementById('workshop-selector').classList.remove('open');
 | 
			
		||||
    document.body.classList.remove('modal-open');
 | 
			
		||||
  }
 | 
			
		||||
  document.getElementById('workshop-selector').addEventListener('click', function(event) {
 | 
			
		||||
    if (event.target.id == 'workshop-selector') {
 | 
			
		||||
      closeWorkshopSelector();
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
  function _post(form, params, f) {
 | 
			
		||||
    var request = new XMLHttpRequest();
 | 
			
		||||
    request.onreadystatechange = function() {
 | 
			
		||||
      if (request.readyState == 4) {
 | 
			
		||||
        if (request.status == 200) {
 | 
			
		||||
          f(request.responseText);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    request.open('POST', form.getAttribute('action'), true);
 | 
			
		||||
    request.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded; charset=UTF-8');
 | 
			
		||||
    params['authenticity_token'] = form.querySelector('[name="authenticity_token"]').value;
 | 
			
		||||
    var data = [];
 | 
			
		||||
    for (var key in params) {
 | 
			
		||||
      data.push(key + '=' + params[key]);
 | 
			
		||||
    }
 | 
			
		||||
    request.send(data.join('&'));
 | 
			
		||||
  }
 | 
			
		||||
  function selectorMatches(el, selector) {
 | 
			
		||||
    var p = Element.prototype;
 | 
			
		||||
    var f = p.matches || p.webkitMatchesSelector || p.mozMatchesSelector || p.msMatchesSelector || function(s) {
 | 
			
		||||
      return [].indexOf.call(document.querySelectorAll(s), this) !== -1;
 | 
			
		||||
    };
 | 
			
		||||
    return f.call(el, selector);
 | 
			
		||||
  }
 | 
			
		||||
  function updateSchedule(html) {
 | 
			
		||||
    var schedule = document.getElementById('schedule-preview');
 | 
			
		||||
    var s = document.createElement('div');
 | 
			
		||||
    s.innerHTML = html;
 | 
			
		||||
    schedule.innerHTML = s.children[0].innerHTML;
 | 
			
		||||
    schedule.classList.remove('requesting');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
	document.body.addEventListener('submit', function (event) {
 | 
			
		||||
		if (event.target.classList.contains('deschedule-workshop')) {
 | 
			
		||||
			event.preventDefault();
 | 
			
		||||
			var schedule = document.getElementById('schedule-preview');
 | 
			
		||||
			var form = event.target;
 | 
			
		||||
			schedule.classList.add('requesting');
 | 
			
		||||
			_post(
 | 
			
		||||
					form,
 | 
			
		||||
					{
 | 
			
		||||
						id: form.querySelector('[name="id"]').value,
 | 
			
		||||
						button: 'deschedule_workshop'
 | 
			
		||||
					},
 | 
			
		||||
					updateSchedule
 | 
			
		||||
				);
 | 
			
		||||
		}
 | 
			
		||||
	});
 | 
			
		||||
	document.body.addEventListener('click', function (event) {
 | 
			
		||||
		//console.log(event.target);
 | 
			
		||||
  document.body.addEventListener('submit', function (event) {
 | 
			
		||||
    if (event.target.classList.contains('deschedule-workshop')) {
 | 
			
		||||
      event.preventDefault();
 | 
			
		||||
      var schedule = document.getElementById('schedule-preview');
 | 
			
		||||
      var form = event.target;
 | 
			
		||||
      schedule.classList.add('requesting');
 | 
			
		||||
      _post(
 | 
			
		||||
          form,
 | 
			
		||||
          {
 | 
			
		||||
            id: form.querySelector('[name="id"]').value,
 | 
			
		||||
            button: 'deschedule_workshop'
 | 
			
		||||
          },
 | 
			
		||||
          updateSchedule
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
  document.body.addEventListener('click', function (event) {
 | 
			
		||||
    
 | 
			
		||||
		if (selectorMatches(event.target, 'td.workshop.open, td.workshop.open *')) {
 | 
			
		||||
			//event.stopPropagation();
 | 
			
		||||
			var button = event.target;
 | 
			
		||||
			while (button && button.tagName && button.tagName !== 'TD') {
 | 
			
		||||
				button = button.parentElement;
 | 
			
		||||
			}
 | 
			
		||||
    if (selectorMatches(event.target, 'td.workshop.open, td.workshop.open *')) {
 | 
			
		||||
      var button = event.target;
 | 
			
		||||
      while (button && button.tagName && button.tagName !== 'TD') {
 | 
			
		||||
        button = button.parentElement;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
			document.getElementById('workshop-selector').classList.add('open');
 | 
			
		||||
			var table = document.getElementById('table');
 | 
			
		||||
			table.classList.add('loading');
 | 
			
		||||
			document.body.classList.add('modal-open');
 | 
			
		||||
      document.getElementById('workshop-selector').classList.add('open');
 | 
			
		||||
      var table = document.getElementById('table');
 | 
			
		||||
      table.classList.add('loading');
 | 
			
		||||
      document.body.classList.add('modal-open');
 | 
			
		||||
 | 
			
		||||
			var block = button.getAttribute('data-block');
 | 
			
		||||
			var day = button.getAttribute('data-day');
 | 
			
		||||
			var location = button.getAttribute('data-location');
 | 
			
		||||
      var block = button.getAttribute('data-block');
 | 
			
		||||
      var day = button.getAttribute('data-day');
 | 
			
		||||
      var location = button.getAttribute('data-location');
 | 
			
		||||
      var division = button.getAttribute('data-division');
 | 
			
		||||
 | 
			
		||||
			_post(
 | 
			
		||||
					document.getElementById('workshop-table-form'),
 | 
			
		||||
					{
 | 
			
		||||
						block: block,
 | 
			
		||||
						day: day,
 | 
			
		||||
						location: location,
 | 
			
		||||
						button: 'get-workshop-list'
 | 
			
		||||
					},
 | 
			
		||||
					function (response) {
 | 
			
		||||
						var table = document.getElementById('table');
 | 
			
		||||
						table.innerHTML = response;
 | 
			
		||||
						table.classList.remove('loading');
 | 
			
		||||
						forEachElement('tr.selectable', function(row) {
 | 
			
		||||
							row.addEventListener('click', function(event) {
 | 
			
		||||
								var schedule = document.getElementById('schedule-preview');
 | 
			
		||||
								schedule.classList.add('requesting');
 | 
			
		||||
								closeWorkshopSelector();
 | 
			
		||||
								var form = document.getElementById('workshop-table-form');
 | 
			
		||||
								_post(
 | 
			
		||||
										form,
 | 
			
		||||
										{
 | 
			
		||||
											workshop: row.getAttribute('data-workshop'),
 | 
			
		||||
											block: block,
 | 
			
		||||
											day: day,
 | 
			
		||||
											location: form.querySelector('#event_location').value,
 | 
			
		||||
											button: 'set-workshop'
 | 
			
		||||
										},
 | 
			
		||||
										updateSchedule
 | 
			
		||||
									);
 | 
			
		||||
							});
 | 
			
		||||
						}, table);
 | 
			
		||||
					}
 | 
			
		||||
				);
 | 
			
		||||
		}
 | 
			
		||||
	}, true);
 | 
			
		||||
      _post(
 | 
			
		||||
          document.getElementById('workshop-table-form'),
 | 
			
		||||
          {
 | 
			
		||||
            block: block,
 | 
			
		||||
            day: day,
 | 
			
		||||
            location: location,
 | 
			
		||||
            division: division,
 | 
			
		||||
            button: 'get-workshop-list'
 | 
			
		||||
          },
 | 
			
		||||
          function (response) {
 | 
			
		||||
            var table = document.getElementById('table');
 | 
			
		||||
            table.innerHTML = response;
 | 
			
		||||
            table.classList.remove('loading');
 | 
			
		||||
            forEachElement('tr.selectable', function(row) {
 | 
			
		||||
              row.addEventListener('click', function(event) {
 | 
			
		||||
                var schedule = document.getElementById('schedule-preview');
 | 
			
		||||
                schedule.classList.add('requesting');
 | 
			
		||||
                closeWorkshopSelector();
 | 
			
		||||
                var form = document.getElementById('workshop-table-form');
 | 
			
		||||
                _post(
 | 
			
		||||
                    form,
 | 
			
		||||
                    {
 | 
			
		||||
                      workshop: row.getAttribute('data-workshop'),
 | 
			
		||||
                      block: block,
 | 
			
		||||
                      day: day,
 | 
			
		||||
                      location: form.querySelector('#event_location').value,
 | 
			
		||||
                      button: 'set-workshop'
 | 
			
		||||
                    },
 | 
			
		||||
                    updateSchedule
 | 
			
		||||
                  );
 | 
			
		||||
              });
 | 
			
		||||
            }, table);
 | 
			
		||||
          }
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
  }, true);
 | 
			
		||||
})();
 | 
			
		||||
 | 
			
		||||
@ -1370,7 +1370,6 @@ nav.sub-menu {
 | 
			
		||||
            .event-detail-link {
 | 
			
		||||
                width: auto;
 | 
			
		||||
                font-size: 1.25em;
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
            .event-detail-link, .details, .title {
 | 
			
		||||
                display: inline;
 | 
			
		||||
@ -1403,6 +1402,10 @@ nav.sub-menu {
 | 
			
		||||
                width: 15em;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            li {
 | 
			
		||||
                white-space: normal;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            + .event-detail-link {
 | 
			
		||||
                padding-right: 1em;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
@ -1,5 +1,5 @@
 | 
			
		||||
class ApplicationController < BaseController
 | 
			
		||||
  protect_from_forgery with: :exception, :except => [:do_confirm, :js_error, :admin_update]
 | 
			
		||||
  protect_from_forgery with: :exception, except: [:do_confirm, :js_error, :admin_update]
 | 
			
		||||
 | 
			
		||||
  before_filter :application_setup
 | 
			
		||||
  after_filter  :capture_page_info
 | 
			
		||||
@ -392,13 +392,13 @@ class ApplicationController < BaseController
 | 
			
		||||
  def get_scheule_data(do_analyze = true)
 | 
			
		||||
    conference = @this_conference || @conference
 | 
			
		||||
    @meals = Hash[(conference.meals || {}).map{ |k, v| [k.to_i, v] }].sort.to_h
 | 
			
		||||
    @events = Event.where(:conference_id => conference.id).order(start_time: :asc)
 | 
			
		||||
    @workshops = Workshop.where(:conference_id => conference.id).order(start_time: :asc)
 | 
			
		||||
    @events = Event.where(conference_id: conference.id).order(start_time: :asc)
 | 
			
		||||
    @workshops = Workshop.where(conference_id: conference.id).order(start_time: :asc)
 | 
			
		||||
    @locations = {}
 | 
			
		||||
 | 
			
		||||
    get_block_data
 | 
			
		||||
 | 
			
		||||
    @schedule = {}
 | 
			
		||||
    schedule = {}
 | 
			
		||||
    day_1 = conference.start_date.wday
 | 
			
		||||
 | 
			
		||||
    @workshop_blocks.each_with_index do |info, block|
 | 
			
		||||
@ -407,11 +407,11 @@ class ApplicationController < BaseController
 | 
			
		||||
        day_diff += 7 if day_diff < 0
 | 
			
		||||
        day = (conference.start_date + day_diff.days).to_date
 | 
			
		||||
        time = info['time'].to_f
 | 
			
		||||
        @schedule[day] ||= { times: {}, locations: {} }
 | 
			
		||||
        @schedule[day][:times][time] ||= {}
 | 
			
		||||
        @schedule[day][:times][time][:type] = :workshop
 | 
			
		||||
        @schedule[day][:times][time][:length] = info['length'].to_f
 | 
			
		||||
        @schedule[day][:times][time][:item] = { block: block, workshops: {} }
 | 
			
		||||
        schedule[day] ||= { times: {}, locations: {} }
 | 
			
		||||
        schedule[day][:times][time] ||= {}
 | 
			
		||||
        schedule[day][:times][time][:type] = :workshop
 | 
			
		||||
        schedule[day][:times][time][:length] = info['length'].to_f
 | 
			
		||||
        schedule[day][:times][time][:item] = { block: block, workshops: {} }
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
@ -423,9 +423,9 @@ class ApplicationController < BaseController
 | 
			
		||||
        day_diff += 7 if day_diff < 0
 | 
			
		||||
        day = (conference.start_date + day_diff.days).to_date
 | 
			
		||||
 | 
			
		||||
        if block.present? && @schedule[day].present? && @schedule[day][:times].present? && @schedule[day][:times][block['time'].to_f].present?
 | 
			
		||||
          @schedule[day][:times][block['time'].to_f][:item][:workshops][workshop.event_location_id] = { workshop: workshop, status: { errors: [], warnings: [], conflict_score: nil } }
 | 
			
		||||
          @schedule[day][:locations][workshop.event_location_id] ||= workshop.event_location if workshop.event_location.present?
 | 
			
		||||
        if block.present? && schedule[day].present? && schedule[day][:times].present? && schedule[day][:times][block['time'].to_f].present?
 | 
			
		||||
          schedule[day][:times][block['time'].to_f][:item][:workshops][workshop.event_location_id] = { workshop: workshop, status: { errors: [], warnings: [], conflict_score: nil } }
 | 
			
		||||
          schedule[day][:locations][workshop.event_location_id] ||= workshop.event_location if workshop.event_location.present?
 | 
			
		||||
        end
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
@ -433,50 +433,50 @@ class ApplicationController < BaseController
 | 
			
		||||
    @meals.each do |time, meal|
 | 
			
		||||
      day = meal['day'].to_date
 | 
			
		||||
      time = meal['time'].to_f
 | 
			
		||||
      @schedule[day] ||= {}
 | 
			
		||||
      @schedule[day][:times] ||= {}
 | 
			
		||||
      @schedule[day][:times][time] ||= {}
 | 
			
		||||
      @schedule[day][:times][time][:type] = :meal
 | 
			
		||||
      @schedule[day][:times][time][:length] = (meal['length'] || 1.0).to_f
 | 
			
		||||
      @schedule[day][:times][time][:item] = meal
 | 
			
		||||
      schedule[day] ||= {}
 | 
			
		||||
      schedule[day][:times] ||= {}
 | 
			
		||||
      schedule[day][:times][time] ||= {}
 | 
			
		||||
      schedule[day][:times][time][:type] = :meal
 | 
			
		||||
      schedule[day][:times][time][:length] = (meal['length'] || 1.0).to_f
 | 
			
		||||
      schedule[day][:times][time][:item] = meal
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    @events.each do |event|
 | 
			
		||||
      if event.present? && event.start_time.present? && event.end_time.present?
 | 
			
		||||
        day = event.start_time.midnight.to_date
 | 
			
		||||
        time = event.start_time.hour.to_f + (event.start_time.min / 60.0)
 | 
			
		||||
        @schedule[day] ||= {}
 | 
			
		||||
        @schedule[day][:times] ||= {}
 | 
			
		||||
        @schedule[day][:times][time] ||= {}
 | 
			
		||||
        @schedule[day][:times][time][:type] = :event
 | 
			
		||||
        @schedule[day][:times][time][:length] = (event.end_time - event.start_time) / 3600.0
 | 
			
		||||
        @schedule[day][:times][time][:item] = event
 | 
			
		||||
        schedule[day] ||= {}
 | 
			
		||||
        schedule[day][:times] ||= {}
 | 
			
		||||
        schedule[day][:times][time] ||= {}
 | 
			
		||||
        schedule[day][:times][time][:type] = :event
 | 
			
		||||
        schedule[day][:times][time][:length] = (event.end_time - event.start_time) / 3600.0
 | 
			
		||||
        schedule[day][:times][time][:item] = event
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    @schedule = @schedule.sort.to_h
 | 
			
		||||
    @schedule.each do |day, data|
 | 
			
		||||
      @schedule[day][:times] = data[:times].sort.to_h
 | 
			
		||||
    schedule = schedule.sort.to_h
 | 
			
		||||
    schedule.each do |day, data|
 | 
			
		||||
      schedule[day][:times] = data[:times].sort.to_h
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    @schedule.each do |day, data|
 | 
			
		||||
    schedule.each do |day, data|
 | 
			
		||||
      last_event = nil
 | 
			
		||||
      data[:times].each do |time, time_data|
 | 
			
		||||
        if last_event.present?
 | 
			
		||||
          @schedule[day][:times][last_event][:next_event] = time
 | 
			
		||||
          schedule[day][:times][last_event][:next_event] = time
 | 
			
		||||
        end
 | 
			
		||||
        last_event = time
 | 
			
		||||
      end
 | 
			
		||||
      @schedule[day][:num_locations] = (data[:locations] || []).size
 | 
			
		||||
      schedule[day][:num_locations] = (data[:locations] || []).size
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    @schedule.deep_dup.each do |day, data|
 | 
			
		||||
    schedule.deep_dup.each do |day, data|
 | 
			
		||||
      data[:times].each do |time, time_data|
 | 
			
		||||
        if time_data[:next_event].present? || time_data[:length] > @this_conference.schedule_interval
 | 
			
		||||
          span = @this_conference.schedule_interval
 | 
			
		||||
          length = time_data[:next_event].present? ? time_data[:next_event] - time : time_data[:length]
 | 
			
		||||
          while span < length
 | 
			
		||||
            @schedule[day][:times][time + span] ||= {
 | 
			
		||||
            schedule[day][:times][time + span] ||= {
 | 
			
		||||
              type: (span >= time_data[:length] ? :empty : :nil),
 | 
			
		||||
              length: @this_conference.schedule_interval
 | 
			
		||||
            }
 | 
			
		||||
@ -486,16 +486,20 @@ class ApplicationController < BaseController
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    @schedule = @schedule.sort.to_h
 | 
			
		||||
    # schedule = schedule.sort.to_h
 | 
			
		||||
 | 
			
		||||
    @schedule.each do |day, data|
 | 
			
		||||
      @schedule[day][:times] = data[:times].sort.to_h
 | 
			
		||||
      @schedule[day][:locations] ||= {}
 | 
			
		||||
    schedule.each do |day, data|
 | 
			
		||||
      # @schedule[day] = [{}]
 | 
			
		||||
      # division = 0
 | 
			
		||||
      # schedule[day][:num_locations] = schedule[day][:num_locations]
 | 
			
		||||
      # schedule[day][:times] = data[:times].sort.to_h
 | 
			
		||||
      schedule[day][:times] = data[:times].sort.to_h
 | 
			
		||||
      schedule[day][:locations] ||= {}
 | 
			
		||||
      # # sort the locations by name
 | 
			
		||||
      schedule[day][:locations] = schedule[day][:locations].sort_by { |event_id, event| event.present? ? event.title.downcase   : '' }.to_h
 | 
			
		||||
 | 
			
		||||
      # sort the locations by name      
 | 
			
		||||
      @schedule[day][:locations] = @schedule[day][:locations].sort_by { |event_id, event| event.present? ? event.title.downcase   : '' }.to_h
 | 
			
		||||
      # add an empty block if no workshops are scheduled on this day yet
 | 
			
		||||
      @schedule[day][:locations][0] = :add if do_analyze || @schedule[day][:locations].empty?
 | 
			
		||||
      # # add an empty block if no workshops are scheduled on this day yet
 | 
			
		||||
      # schedule[day][:locations][0] = :add if do_analyze || schedule[day][:locations].empty?
 | 
			
		||||
 | 
			
		||||
      if do_analyze
 | 
			
		||||
        data[:times].each do |time, time_data|
 | 
			
		||||
@ -512,8 +516,8 @@ class ApplicationController < BaseController
 | 
			
		||||
                    workshop_i.active_facilitators.each do |facilitator_i|
 | 
			
		||||
                      workshop_j.active_facilitators.each do |facilitator_j|
 | 
			
		||||
                        if facilitator_i.id == facilitator_j.id
 | 
			
		||||
                          @schedule[day][:times][time][:status] ||= {}
 | 
			
		||||
                          @schedule[day][:times][time][:item][:workshops][ids[j]][:status][:errors] << {
 | 
			
		||||
                          schedule[day][:times][time][:status] ||= {}
 | 
			
		||||
                          schedule[day][:times][time][:item][:workshops][ids[j]][:status][:errors] << {
 | 
			
		||||
                              name: :common_facilitator,
 | 
			
		||||
                              facilitator: facilitator_i,
 | 
			
		||||
                              workshop: workshop_i,
 | 
			
		||||
@ -533,7 +537,7 @@ class ApplicationController < BaseController
 | 
			
		||||
                amenities = JSON.parse(location.amenities || '[]').map &:to_sym
 | 
			
		||||
 | 
			
		||||
                needs.each do |need|
 | 
			
		||||
                  @schedule[day][:times][time][:item][:workshops][ids[i]][:status][:errors] << {
 | 
			
		||||
                  schedule[day][:times][time][:item][:workshops][ids[i]][:status][:errors] << {
 | 
			
		||||
                      name: :need_not_available,
 | 
			
		||||
                      need: need,
 | 
			
		||||
                      location: location,
 | 
			
		||||
@ -555,13 +559,55 @@ class ApplicationController < BaseController
 | 
			
		||||
                  end
 | 
			
		||||
                end
 | 
			
		||||
 | 
			
		||||
                @schedule[day][:times][time][:item][:workshops][ids[i]][:status][:conflict_score] = (interests & (workshop_i.interested.map { | u | u.user_id })).length
 | 
			
		||||
                schedule[day][:times][time][:item][:workshops][ids[i]][:status][:conflict_score] = (interests & (workshop_i.interested.map { | u | u.user_id })).length
 | 
			
		||||
              end
 | 
			
		||||
            end
 | 
			
		||||
          end
 | 
			
		||||
        end
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    @schedule = {}
 | 
			
		||||
    schedule.sort.to_h.each do |day, data|
 | 
			
		||||
      @schedule[day] = []
 | 
			
		||||
      division = 0
 | 
			
		||||
      # @schedule[day][division] = data
 | 
			
		||||
      locations = nil
 | 
			
		||||
      @schedule[day][division] = {}
 | 
			
		||||
      @schedule[day][division][:times] = {}
 | 
			
		||||
      # @schedule[day][division][:times] = data[:times]
 | 
			
		||||
 | 
			
		||||
      # # sort the locations by name
 | 
			
		||||
      # @schedule[day][:locations] = schedule[day][:locations].sort_by { |event_id, event| event.present? ? event.title.downcase   : '' }.to_h
 | 
			
		||||
 | 
			
		||||
      # # add an empty block if no workshops are scheduled on this day yet
 | 
			
		||||
      # schedule[day][:locations][0] = :add if do_analyze || schedule[day][:locations].empty?
 | 
			
		||||
 | 
			
		||||
      # last_time_data = nil
 | 
			
		||||
      data[:times].each do |time, time_data|
 | 
			
		||||
        if time_data[:type] == :workshop && time_data[:item].present? && time_data[:item][:workshops].present?
 | 
			
		||||
          if !locations.nil? && ((locations.keys - time_data[:item][:workshops].keys) | (time_data[:item][:workshops].keys - locations.keys)).length > 0
 | 
			
		||||
            # data[:locations]
 | 
			
		||||
            # xxx
 | 
			
		||||
            @schedule[day][division][:locations] = locations.deep_dup
 | 
			
		||||
            @schedule[day][division][:locations][0] = :add if do_analyze || locations.empty?
 | 
			
		||||
            locations = data[:locations].select { |id, l| time_data[:item][:workshops][id].present? }
 | 
			
		||||
 | 
			
		||||
            division += 1
 | 
			
		||||
            @schedule[day][division] = {}
 | 
			
		||||
            @schedule[day][division][:times] = {}
 | 
			
		||||
          else
 | 
			
		||||
            locations = data[:locations].select { |id, l| time_data[:item][:workshops][id].present? }
 | 
			
		||||
          end
 | 
			
		||||
        end
 | 
			
		||||
        # last_time_data = time_data
 | 
			
		||||
        @schedule[day][division][:times][time] = time_data
 | 
			
		||||
      end
 | 
			
		||||
      locations ||= data[:locations]
 | 
			
		||||
      @schedule[day][division][:locations] = locations
 | 
			
		||||
      @schedule[day][division][:locations][0] = :add if do_analyze || locations.empty?
 | 
			
		||||
      @schedule[day][division][:num_locations] = locations.length
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  protected
 | 
			
		||||
 | 
			
		||||
@ -253,46 +253,7 @@ class ConferenceAdministrationController < ApplicationController
 | 
			
		||||
          format.xlsx { render xlsx: '../conferences/stats', filename: "stats-#{DateTime.now.strftime('%Y-%m-%d')}" }
 | 
			
		||||
        end
 | 
			
		||||
      else
 | 
			
		||||
        if params[:sort_column]
 | 
			
		||||
          col = params[:sort_column].to_sym
 | 
			
		||||
          @excel_data[:data].sort_by! do |row|
 | 
			
		||||
            value = row[col]
 | 
			
		||||
 | 
			
		||||
            if row[:raw_values].key?(col)
 | 
			
		||||
              value = if row[:raw_values][col].is_a?(TrueClass)
 | 
			
		||||
                        't'
 | 
			
		||||
                      elsif row[:raw_values][col].is_a?(FalseClass)
 | 
			
		||||
                        ''
 | 
			
		||||
                      else
 | 
			
		||||
                        row[:raw_values][col]
 | 
			
		||||
                      end
 | 
			
		||||
            elsif value.is_a?(City)
 | 
			
		||||
              value = value.sortable_string
 | 
			
		||||
            end
 | 
			
		||||
 | 
			
		||||
            if value.nil?
 | 
			
		||||
              case @excel_data[:column_types][col]
 | 
			
		||||
              when :datetime, [:date, :day]
 | 
			
		||||
                value = Date.new
 | 
			
		||||
              when :money
 | 
			
		||||
                value = 0
 | 
			
		||||
              else
 | 
			
		||||
                value = ''
 | 
			
		||||
              end
 | 
			
		||||
            end
 | 
			
		||||
 | 
			
		||||
            value
 | 
			
		||||
          end
 | 
			
		||||
 | 
			
		||||
          if params[:sort_dir] == 'up'
 | 
			
		||||
            @sort_dir = :up
 | 
			
		||||
            @excel_data[:data].reverse!
 | 
			
		||||
          end
 | 
			
		||||
 | 
			
		||||
          @sort_column = col
 | 
			
		||||
        else
 | 
			
		||||
          @sort_column = :name
 | 
			
		||||
        end
 | 
			
		||||
        sort_data(params[:sort_column], params[:sort_dir], :name)
 | 
			
		||||
      end
 | 
			
		||||
 | 
			
		||||
      @registration_count = @registrations.size
 | 
			
		||||
@ -324,6 +285,22 @@ class ConferenceAdministrationController < ApplicationController
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    def administrate_workshops
 | 
			
		||||
      get_workshops(true)
 | 
			
		||||
      if request.format.xlsx?
 | 
			
		||||
        logger.info "Generating stats.xls"
 | 
			
		||||
        return respond_to do |format|
 | 
			
		||||
          format.xlsx { render xlsx: '../conferences/stats', filename: "workshops-#{DateTime.now.strftime('%Y-%m-%d')}" }
 | 
			
		||||
        end
 | 
			
		||||
      else
 | 
			
		||||
        sort_data(params[:sort_column], params[:sort_dir], :name)
 | 
			
		||||
      end
 | 
			
		||||
 | 
			
		||||
      if request.xhr?
 | 
			
		||||
        render html: view_context.html_table(@excel_data, view_context.registrations_table_options)
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    def administrate_check_in
 | 
			
		||||
      sort_weight = {
 | 
			
		||||
        checked_in: 5,
 | 
			
		||||
@ -769,6 +746,179 @@ class ConferenceAdministrationController < ApplicationController
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    def get_workshops(html_format = false, id = nil, conference = @this_conference)
 | 
			
		||||
      @workshops = conference.workshops.sort_by { |w| w.title.downcase }
 | 
			
		||||
      @excel_data = {
 | 
			
		||||
        columns: [
 | 
			
		||||
            :title,
 | 
			
		||||
            :owner,
 | 
			
		||||
            :locale,
 | 
			
		||||
            :date,
 | 
			
		||||
            :info,
 | 
			
		||||
            :notes,
 | 
			
		||||
            :facilitators
 | 
			
		||||
          ] + User.AVAILABLE_LANGUAGES.map { |l| "language_#{l}".to_sym } +
 | 
			
		||||
          Workshop.all_needs.map { |n| "need_#{n}".to_sym } + [
 | 
			
		||||
            :theme,
 | 
			
		||||
            :space
 | 
			
		||||
          ] + (User.AVAILABLE_LANGUAGES - [I18n.locale]).map { |l| "title_#{l}".to_sym } +
 | 
			
		||||
          (User.AVAILABLE_LANGUAGES - [I18n.locale]).map { |l| "info_#{l}".to_sym },
 | 
			
		||||
        column_types: {
 | 
			
		||||
            title: :bold,
 | 
			
		||||
            date: :datetime,
 | 
			
		||||
            info: :text,
 | 
			
		||||
            notes: :text,
 | 
			
		||||
            owner: :email
 | 
			
		||||
          },
 | 
			
		||||
        keys: {
 | 
			
		||||
            title: 'forms.labels.generic.title',
 | 
			
		||||
            owner: 'roles.workshops.facilitator.creator',
 | 
			
		||||
            locale: 'articles.conference_registration.terms.Preferred_Languages',
 | 
			
		||||
            info: 'forms.labels.generic.info',
 | 
			
		||||
            date: 'workshop.created_at',
 | 
			
		||||
            notes: 'forms.labels.generic.notes',
 | 
			
		||||
            facilitators: 'roles.workshops.facilitator.facilitator',
 | 
			
		||||
            theme: 'articles.workshops.headings.theme',
 | 
			
		||||
            space: 'articles.workshops.headings.space'
 | 
			
		||||
          },
 | 
			
		||||
        data: []
 | 
			
		||||
      }
 | 
			
		||||
      @excel_data[:key_vars] = {}
 | 
			
		||||
      User.AVAILABLE_LANGUAGES.each do |l|
 | 
			
		||||
        @excel_data[:keys]["language_#{l}".to_sym] = "languages.#{l}"
 | 
			
		||||
        if l != I18n.locale
 | 
			
		||||
          @excel_data[:keys]["title_#{l}".to_sym] = 'translate.content.item_translation'
 | 
			
		||||
          @excel_data[:key_vars]["title_#{l}".to_sym] = { language: view_context.language_name(l), item: I18n.t('forms.labels.generic.title') }
 | 
			
		||||
 | 
			
		||||
          @excel_data[:keys]["info_#{l}".to_sym] = 'translate.content.item_translation'
 | 
			
		||||
          @excel_data[:key_vars]["info_#{l}".to_sym] = { language: view_context.language_name(l), item: I18n.t('forms.labels.generic.info') }
 | 
			
		||||
          @excel_data[:column_types]["info_#{l}".to_sym] = :text
 | 
			
		||||
        end
 | 
			
		||||
      end
 | 
			
		||||
 | 
			
		||||
      Workshop.all_needs.each do |n|
 | 
			
		||||
        @excel_data[:keys]["need_#{n}".to_sym] = "workshop.options.needs.#{n}"
 | 
			
		||||
      end
 | 
			
		||||
 | 
			
		||||
      @workshops.each do |w|
 | 
			
		||||
        if w.present?
 | 
			
		||||
          if id.nil? || id == w.id
 | 
			
		||||
            owner = User.find(w.creator)
 | 
			
		||||
            facilitators = w.collaborators.map { |f| User.find(f) }
 | 
			
		||||
            data = {
 | 
			
		||||
              id: w.id,
 | 
			
		||||
              title: w.title,
 | 
			
		||||
              info: view_context.strip_tags(w.info),
 | 
			
		||||
              notes: view_context.strip_tags(w.notes),
 | 
			
		||||
              owner: owner.name,
 | 
			
		||||
              locale: w.locale.present? ? (view_context.language_name w.locale) : '',
 | 
			
		||||
              date: w.created_at ? w.created_at.strftime("%F %T") : '',
 | 
			
		||||
              facilitators: facilitators.map { |f| f.name }.join(', '),
 | 
			
		||||
              theme: w.theme && Workshop.all_themes.include?(w.theme.to_sym) ? I18n.t("workshop.options.theme.#{w.theme}") : w.theme,
 | 
			
		||||
              space: w.space && Workshop.all_spaces.include?(w.space.to_sym) ? I18n.t("workshop.options.space.#{w.space}") : '',
 | 
			
		||||
              raw_values: {
 | 
			
		||||
                info: w.info,
 | 
			
		||||
                owner: owner.email,
 | 
			
		||||
                notes: w.notes,
 | 
			
		||||
                locale: w.locale,
 | 
			
		||||
                facilitators: facilitators.map { |f| f.email }.join(', '),
 | 
			
		||||
                theme: w.theme,
 | 
			
		||||
                space: w.space
 | 
			
		||||
              },
 | 
			
		||||
              html_values: {
 | 
			
		||||
              }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            languages = JSON.parse(w.languages || '[]').map &:to_sym
 | 
			
		||||
            User.AVAILABLE_LANGUAGES.each do |l|
 | 
			
		||||
              in_language = ((languages || []).include? l.to_sym)
 | 
			
		||||
              data["language_#{l}".to_sym] = (in_language ? I18n.t('articles.conference_registration.questions.bike.yes') : '')
 | 
			
		||||
              data[:raw_values]["language_#{l}".to_sym] = in_language
 | 
			
		||||
 | 
			
		||||
              if l != I18n.locale
 | 
			
		||||
                data["title_#{l}".to_sym] = w.get_column_for_locale!(:title, l, false)
 | 
			
		||||
                data["info_#{l}".to_sym] = view_context.strip_tags(w.get_column_for_locale!(:info, l, false))
 | 
			
		||||
                data[:raw_values]["info_#{l}".to_sym] = w.get_column_for_locale!(:info, l, false)
 | 
			
		||||
              end
 | 
			
		||||
            end
 | 
			
		||||
 | 
			
		||||
            needs = JSON.parse(w.needs || '[]').map &:to_sym
 | 
			
		||||
            Workshop.all_needs.each do |n|
 | 
			
		||||
              in_need = ((needs || []).include? n.to_sym)
 | 
			
		||||
              data["need_#{n}".to_sym] = (in_need ? I18n.t('articles.conference_registration.questions.bike.yes') : '')
 | 
			
		||||
              data[:raw_values]["need_#{n}".to_sym] = in_need
 | 
			
		||||
            end
 | 
			
		||||
 | 
			
		||||
            @excel_data[:data] << data
 | 
			
		||||
          end
 | 
			
		||||
        end
 | 
			
		||||
      end
 | 
			
		||||
 | 
			
		||||
      if html_format
 | 
			
		||||
        @column_options = {
 | 
			
		||||
          locale: I18n.backend.enabled_locales.map { |l| [(view_context.language_name l), l] },
 | 
			
		||||
          theme: Workshop.all_themes.map { |t| [I18n.t("workshop.options.theme.#{t}"), t] },
 | 
			
		||||
          space: Workshop.all_spaces.map { |s| [I18n.t("workshop.options.space.#{s}"), s] }
 | 
			
		||||
        }
 | 
			
		||||
        @column_options[:theme] += ((conference.workshops.map { |w| w.theme }) - Workshop.all_themes.map(&:to_s)).uniq.map { |t| [t, t] }
 | 
			
		||||
        User.AVAILABLE_LANGUAGES.each do |l|
 | 
			
		||||
          @column_options["language_#{l}".to_sym] = [
 | 
			
		||||
              [I18n.t("articles.conference_registration.questions.bike.yes"), true]
 | 
			
		||||
            ]
 | 
			
		||||
        end
 | 
			
		||||
        Workshop.all_needs.each do |n|
 | 
			
		||||
          @column_options["need_#{n}".to_sym] = [
 | 
			
		||||
              [I18n.t("articles.conference_registration.questions.bike.yes"), true]
 | 
			
		||||
            ]
 | 
			
		||||
        end
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    def sort_data(col, sort_dir, default_col)
 | 
			
		||||
      if col
 | 
			
		||||
        col = col.to_sym
 | 
			
		||||
        @excel_data[:data].sort_by! do |row|
 | 
			
		||||
          value = row[col]
 | 
			
		||||
 | 
			
		||||
          if row[:raw_values].key?(col)
 | 
			
		||||
            value = if row[:raw_values][col].is_a?(TrueClass)
 | 
			
		||||
                      't'
 | 
			
		||||
                    elsif row[:raw_values][col].is_a?(FalseClass)
 | 
			
		||||
                      ''
 | 
			
		||||
                    elsif @excel_data[:column_types][col] == :text
 | 
			
		||||
                      view_context.strip_tags(row[:raw_values][col] || '').downcase
 | 
			
		||||
                    else
 | 
			
		||||
                      row[:raw_values][col]
 | 
			
		||||
                    end
 | 
			
		||||
          elsif value.is_a?(City)
 | 
			
		||||
            value = value.sortable_string
 | 
			
		||||
          end
 | 
			
		||||
 | 
			
		||||
          if value.nil?
 | 
			
		||||
            case @excel_data[:column_types][col]
 | 
			
		||||
            when :datetime, [:date, :day]
 | 
			
		||||
              value = Date.new
 | 
			
		||||
            when :money
 | 
			
		||||
              value = 0
 | 
			
		||||
            else
 | 
			
		||||
              value = ''
 | 
			
		||||
            end
 | 
			
		||||
          end
 | 
			
		||||
 | 
			
		||||
          value
 | 
			
		||||
        end
 | 
			
		||||
 | 
			
		||||
        if sort_dir == 'up'
 | 
			
		||||
          @sort_dir = :up
 | 
			
		||||
          @excel_data[:data].reverse!
 | 
			
		||||
        end
 | 
			
		||||
 | 
			
		||||
        @sort_column = col
 | 
			
		||||
      else
 | 
			
		||||
        @sort_column = default_col
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    def get_housing_data
 | 
			
		||||
      @hosts = {}
 | 
			
		||||
      @guests = {}
 | 
			
		||||
@ -1255,6 +1405,85 @@ class ConferenceAdministrationController < ApplicationController
 | 
			
		||||
      return nil
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    def admin_update_workshops
 | 
			
		||||
      if params[:button] == 'update'
 | 
			
		||||
        workshop = Workshop.where(
 | 
			
		||||
              id: params[:key].to_i,
 | 
			
		||||
              conference_id: @this_conference.id
 | 
			
		||||
            ).limit(1).first
 | 
			
		||||
 | 
			
		||||
        params.each do |key, value|
 | 
			
		||||
          case key.to_sym
 | 
			
		||||
          when :owner
 | 
			
		||||
            user = User.get(value.strip)
 | 
			
		||||
            user_role = WorkshopFacilitator.where(user_id: user.id, workshop_id: workshop.id).first || WorkshopFacilitator.new(user_id: user.id, workshop_id: workshop.id)
 | 
			
		||||
            owner_role = WorkshopFacilitator.where(role: :creator, workshop_id: workshop.id).first
 | 
			
		||||
            if !owner_role || owner_role.user_id != user.id
 | 
			
		||||
              owner_role.role = :collaborator
 | 
			
		||||
              user_role.role = :creator
 | 
			
		||||
              owner_role.save!
 | 
			
		||||
              user_role.save!
 | 
			
		||||
            end
 | 
			
		||||
          when :facilitators
 | 
			
		||||
            ids = []
 | 
			
		||||
            value.split(/[\s,;]+/).each do |email|
 | 
			
		||||
              user = User.get(email)
 | 
			
		||||
              ids << user.id
 | 
			
		||||
              user_role = WorkshopFacilitator.where(user_id: user.id, workshop_id: workshop.id).first || WorkshopFacilitator.new(user_id: user.id, workshop_id: workshop.id)
 | 
			
		||||
              unless user_role.role == 'creator' || user_role.role == 'collaborator'
 | 
			
		||||
                user_role.role = 'collaborator'
 | 
			
		||||
                user_role.save
 | 
			
		||||
              end
 | 
			
		||||
            end
 | 
			
		||||
            WorkshopFacilitator.where("workshop_id = ? AND role = ? AND user_id NOT IN (?)", workshop.id, 'collaborator', ids).destroy_all
 | 
			
		||||
          when :title, :locale, :date, :info, :notes, :theme, :space
 | 
			
		||||
            workshop.send("#{key}=", value.present? ? value : nil)
 | 
			
		||||
          else
 | 
			
		||||
            if key.start_with?('language_')
 | 
			
		||||
              l = key.split('_').last.to_sym
 | 
			
		||||
              languages = JSON.parse(workshop.languages || '[]').map &:to_sym
 | 
			
		||||
              if User.AVAILABLE_LANGUAGES.include? l
 | 
			
		||||
                if value.present?
 | 
			
		||||
                  languages |= [l]
 | 
			
		||||
                else
 | 
			
		||||
                  languages -= [l]
 | 
			
		||||
                end
 | 
			
		||||
                workshop.languages = languages.to_json
 | 
			
		||||
              end
 | 
			
		||||
            elsif key.start_with?('need_')
 | 
			
		||||
              n = key.split('_').last.to_sym
 | 
			
		||||
              needs = JSON.parse(workshop.needs || '[]').map &:to_sym
 | 
			
		||||
              if Workshop.all_needs.include? n
 | 
			
		||||
                if value.present?
 | 
			
		||||
                  needs |= [n]
 | 
			
		||||
                else
 | 
			
		||||
                  needs -= [n]
 | 
			
		||||
                end
 | 
			
		||||
                workshop.needs = needs.to_json
 | 
			
		||||
              end
 | 
			
		||||
            elsif key.start_with?('title_')
 | 
			
		||||
              l = key.split('_').last.to_sym
 | 
			
		||||
              workshop.set_column_for_locale(:title, l, value)
 | 
			
		||||
            elsif key.start_with?('info_')
 | 
			
		||||
              l = key.split('_').last.to_sym
 | 
			
		||||
              workshop.set_column_for_locale(:info, l, value)
 | 
			
		||||
            end
 | 
			
		||||
          end
 | 
			
		||||
        end
 | 
			
		||||
        workshop.save!
 | 
			
		||||
 | 
			
		||||
        get_workshops(true, params[:key].to_i) 
 | 
			
		||||
        options = view_context.workshops_table_options
 | 
			
		||||
        options[:html] = true
 | 
			
		||||
        
 | 
			
		||||
        render html: view_context.excel_rows(@excel_data, {}, options)
 | 
			
		||||
      else
 | 
			
		||||
        do_404
 | 
			
		||||
      end
 | 
			
		||||
 | 
			
		||||
      return nil
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    def admin_update_check_in
 | 
			
		||||
      unless params[:button] == 'cancel'
 | 
			
		||||
        user_id = params[:user_id]
 | 
			
		||||
@ -1478,7 +1707,7 @@ class ConferenceAdministrationController < ApplicationController
 | 
			
		||||
        end
 | 
			
		||||
 | 
			
		||||
        (params[:title] || {}).each do |locale, value|
 | 
			
		||||
          event.set_column_for_locale(:title, locale, html_value(value), current_user.id) if value != event._title(locale) && view_context.strip_tags(value).strip.present?
 | 
			
		||||
          event.set_column_for_locale(:title, locale, html_value(value), current_user.id) if value != event._title(locale) && view_context.strip.present?
 | 
			
		||||
        end
 | 
			
		||||
 | 
			
		||||
        event.save
 | 
			
		||||
@ -1555,8 +1784,18 @@ class ConferenceAdministrationController < ApplicationController
 | 
			
		||||
        @time = @workshop_blocks[@block]['time'].to_f
 | 
			
		||||
        @day = (Date.parse params[:day])
 | 
			
		||||
        @location = params[:location]
 | 
			
		||||
        @division = params[:division].to_i
 | 
			
		||||
        @event_location = @location.present? && @location.to_i > 0 ? EventLocation.find(@location.to_i) : nil
 | 
			
		||||
 | 
			
		||||
        @schedule ||= {}
 | 
			
		||||
        @schedule[@day] ||= {}
 | 
			
		||||
        @schedule[@day][@division] ||= []
 | 
			
		||||
        @schedule[@day][@division][:times] ||= {}
 | 
			
		||||
        @schedule[@day][@division][:times][@time] ||= {}
 | 
			
		||||
        @schedule[@day][@division][:times][@time][:item] ||= {}
 | 
			
		||||
        @schedule[@day][@division][:times][@time][:item][:workshops] || {}
 | 
			
		||||
        @invalid_locations = @schedule[@day][@division.to_i][:times][@time][:item][:workshops].keys
 | 
			
		||||
 | 
			
		||||
        @workshops.sort { |a, b| a.title.downcase <=> b.title.downcase }.each do |workshop|
 | 
			
		||||
          @ordered_workshops[workshop.id] = workshop
 | 
			
		||||
        end
 | 
			
		||||
 | 
			
		||||
@ -38,7 +38,8 @@ module AdminHelper
 | 
			
		||||
      events: [
 | 
			
		||||
          :locations,
 | 
			
		||||
          :meals,
 | 
			
		||||
          :events
 | 
			
		||||
          :events,
 | 
			
		||||
          :workshops
 | 
			
		||||
        ],
 | 
			
		||||
      schedule: [
 | 
			
		||||
          :workshop_times,
 | 
			
		||||
@ -177,8 +178,8 @@ module AdminHelper
 | 
			
		||||
  def available_dates_match?(host, guest)
 | 
			
		||||
    return false unless host.housing_data['availability'].present? && host.housing_data['availability'][1].present?
 | 
			
		||||
    return false unless guest.arrival.present? && guest.departure.present?
 | 
			
		||||
    if host.housing_data['availability'][0] <= guest.arrival &&
 | 
			
		||||
      host.housing_data['availability'][1] >= guest.departure
 | 
			
		||||
    if host.housing_data['availability'][0].to_date <= guest.arrival.to_date &&
 | 
			
		||||
      host.housing_data['availability'][1].to_date >= guest.departure.to_date
 | 
			
		||||
      return true
 | 
			
		||||
     end
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -11,13 +11,15 @@ module FormHelper
 | 
			
		||||
    # set the selected locale
 | 
			
		||||
    selected_locale = (options[:locale] || object.locale || I18n.locale).to_sym
 | 
			
		||||
 | 
			
		||||
    I18n.backend.enabled_locales.each do |locale|
 | 
			
		||||
    locales = I18n.backend.enabled_locales.map { |l| [l, _("languages.#{l}")] }.sort_by { |l| l.last.downcase }.to_h
 | 
			
		||||
 | 
			
		||||
    locales.each do |locale, locale_name|
 | 
			
		||||
      # ses if this should b the selected field
 | 
			
		||||
      class_name = selected_locale == locale.to_sym ? 'selected' : nil
 | 
			
		||||
 | 
			
		||||
      # add the locale to the nav
 | 
			
		||||
      nav += content_tag(:li,
 | 
			
		||||
          content_tag(:a, _("languages.#{locale}"), href: 'javascript:void(0)'),
 | 
			
		||||
          content_tag(:a, locale_name, href: 'javascript:void(0)'),
 | 
			
		||||
        class: class_name, data: { locale: locale }).html_safe
 | 
			
		||||
 | 
			
		||||
      fields = ''
 | 
			
		||||
 | 
			
		||||
@ -16,7 +16,7 @@ module TableHelper
 | 
			
		||||
          headers = ''
 | 
			
		||||
          options[:column_names].each do |header_name, columns|
 | 
			
		||||
            column_names[header_name] ||= []
 | 
			
		||||
            headers += content_tag(:th, excel_data[:keys][header_name].present? ? _(excel_data[:keys][header_name]) : '', colspan: 2)
 | 
			
		||||
            headers += content_tag(:th, excel_data[:keys][header_name].present? ? I18n.t(excel_data[:keys][header_name], (excel_data[:key_vars] || {})[header_name]) : '', colspan: 2)
 | 
			
		||||
            row_count = columns.size
 | 
			
		||||
            columns.each do |column|
 | 
			
		||||
              column_names[header_name] << column
 | 
			
		||||
@ -43,7 +43,7 @@ module TableHelper
 | 
			
		||||
                  attributes[:rowspan] = options[:row_spans][column]
 | 
			
		||||
                end
 | 
			
		||||
 | 
			
		||||
                column_text = excel_data[:keys][column].present? ? _(excel_data[:keys][column]) : ''
 | 
			
		||||
                column_text = excel_data[:keys][column].present? ? I18n.t(excel_data[:keys][column], (excel_data[:key_vars] || {})[header_name]) : ''
 | 
			
		||||
 | 
			
		||||
                columns_html += content_tag(:th, column_text.html_safe, rowspan: attributes[:rowspan]) + 
 | 
			
		||||
                                edit_column(nil, column, nil, attributes, excel_data, options)
 | 
			
		||||
@ -64,7 +64,7 @@ module TableHelper
 | 
			
		||||
            if (excel_data[:column_types] || {})[column] != :table && ((options[:column_names] || []).include? column)
 | 
			
		||||
              rows += content_tag(:tr, { class: 'always-edit', data: { key: '' } }) do
 | 
			
		||||
                attributes = { class: [excel_data[:column_types][column]], data: { 'column-id' => column } }
 | 
			
		||||
                column_text = excel_data[:keys][column].present? ? _(excel_data[:keys][column]) : ''
 | 
			
		||||
                column_text = excel_data[:keys][column].present? ? I18n.t(excel_data[:keys][column], (excel_data[:key_vars] || {})[header_name]) : ''
 | 
			
		||||
 | 
			
		||||
                columns = content_tag(:th, column_text.html_safe) + edit_column(nil, column, nil, attributes, excel_data, options)
 | 
			
		||||
              end
 | 
			
		||||
@ -124,7 +124,7 @@ module TableHelper
 | 
			
		||||
 | 
			
		||||
    data[:columns].each do |column|
 | 
			
		||||
      unless data[:column_types].present? && data[:column_types][column] == :table
 | 
			
		||||
        column_text = data[:keys][column].present? ? _(data[:keys][column]) : ''
 | 
			
		||||
        column_text = data[:keys][column].present? ? I18n.t(data[:keys][column], (data[:key_vars] || {})[column]) : ''
 | 
			
		||||
        attrs = { class: class_name }
 | 
			
		||||
 | 
			
		||||
        unless @sort_column.nil?
 | 
			
		||||
@ -400,4 +400,29 @@ module TableHelper
 | 
			
		||||
      column_options: @column_options
 | 
			
		||||
    }
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def workshops_table_options
 | 
			
		||||
    {
 | 
			
		||||
      id: 'search-table',
 | 
			
		||||
      class: ['registrations', 'admin-edit'],
 | 
			
		||||
      primary_key: :id,
 | 
			
		||||
      column_names: [
 | 
			
		||||
          :title,
 | 
			
		||||
          :owner,
 | 
			
		||||
          :info,
 | 
			
		||||
          :notes,
 | 
			
		||||
          :locale,
 | 
			
		||||
          :facilitators
 | 
			
		||||
        ] +
 | 
			
		||||
        User.AVAILABLE_LANGUAGES.map { |l| "language_#{l}".to_sym } +
 | 
			
		||||
        (User.AVAILABLE_LANGUAGES - [I18n.locale]).map { |l| "title_#{l}".to_sym } +
 | 
			
		||||
        Workshop.all_needs.map { |n| "need_#{n}".to_sym } + [
 | 
			
		||||
          :theme,
 | 
			
		||||
          :space
 | 
			
		||||
        ],
 | 
			
		||||
      editable: administration_update_path(@this_conference.slug, @admin_step),
 | 
			
		||||
      sortable: administration_step_path(@this_conference.slug, @admin_step),
 | 
			
		||||
      column_options: @column_options
 | 
			
		||||
    }
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
@ -35,7 +35,7 @@
 | 
			
		||||
        = day_select @day, small: true, format: :weekday
 | 
			
		||||
        = hour_select @time, small: true
 | 
			
		||||
        = length_select @length, small: true
 | 
			
		||||
      = translate_fields @event, { title: { type: :textfield, big: true, label: 'forms.labels.generic.title' }, info: { type: :textarea, label: 'forms.labels.generic.info', edit_on: :focus } }
 | 
			
		||||
      = translate_fields @event, { title: { type: :textfield, big: true }, info: { type: :textarea, label: 'forms.labels.generic.info', edit_on: :focus } }
 | 
			
		||||
      .actions.next-prev
 | 
			
		||||
        = button :save, value: :save
 | 
			
		||||
        = button :cancel, value: :cancel, class: :subdued, formnovalidate: true if @event.id.present?
 | 
			
		||||
 | 
			
		||||
@ -6,81 +6,82 @@
 | 
			
		||||
    - else
 | 
			
		||||
      - add_inline_script :schedule if @entire_page
 | 
			
		||||
      #schedule-preview
 | 
			
		||||
        - @schedule.each do |day, data|
 | 
			
		||||
        - @schedule.each do |day, data_array|
 | 
			
		||||
          %h4=date(day, :weekday).html_safe
 | 
			
		||||
          %table.schedule{class: [data[:locations].present? ? 'has-locations' : 'no-locations', "locations-#{data[:num_locations]}"]}
 | 
			
		||||
            - if data[:locations].present? && data[:locations].values.first != :add
 | 
			
		||||
              %thead
 | 
			
		||||
                %tr
 | 
			
		||||
                  %th.corner
 | 
			
		||||
                  - data[:locations].each do |id, location|
 | 
			
		||||
                    %th=location.is_a?(Symbol) ? '' : _!(location.title)
 | 
			
		||||
            %tbody
 | 
			
		||||
              - data[:times].each do |time, time_data|
 | 
			
		||||
                %tr{class: "row-type-#{time_data[:type] || 'nil'}"}
 | 
			
		||||
                  - rowspan = (time_data[:length] * (1 / @this_conference.schedule_interval)).to_i
 | 
			
		||||
                  %th=time(time).html_safe
 | 
			
		||||
                  - if time_data[:type] == :workshop
 | 
			
		||||
          - data_array.each_with_index do |data, division|
 | 
			
		||||
            %table.schedule{class: [data[:locations].present? ? 'has-locations' : 'no-locations', "locations-#{data[:num_locations]}"]}
 | 
			
		||||
              - if data[:locations].present? && data[:locations].values.first != :add
 | 
			
		||||
                %thead
 | 
			
		||||
                  %tr
 | 
			
		||||
                    %th.corner
 | 
			
		||||
                    - data[:locations].each do |id, location|
 | 
			
		||||
                      - if time_data[:item][:workshops][id].present?
 | 
			
		||||
                        - workshop = time_data[:item][:workshops][id][:workshop]
 | 
			
		||||
                        - status = time_data[:item][:workshops][id][:status]
 | 
			
		||||
                      - else
 | 
			
		||||
                        - workshop = status = nil
 | 
			
		||||
                      %td{class: [time_data[:type], workshop.present? ? :filled : :open], rowspan: rowspan, data: workshop.present? ? nil : { block: time_data[:item][:block], day: day, location: id }}
 | 
			
		||||
                        - if workshop.present? && workshop.event_location.present?
 | 
			
		||||
                          .workshop-container
 | 
			
		||||
                            - if @can_edit
 | 
			
		||||
                              -if strip_tags(workshop.notes).strip.present?
 | 
			
		||||
                                = admin_notes(workshop.notes)
 | 
			
		||||
                              - if status[:errors].present?
 | 
			
		||||
                                = admin_status content_tag(:ul, (status[:errors].collect { |error| (_"errors.messages.schedule.#{error[:name].to_s}", vars: error[:i18nVars])}.join).html_safe).html_safe
 | 
			
		||||
                            = link_to view_workshop_path(@conference.slug, workshop.id), class: 'event-detail-link' do
 | 
			
		||||
                              .details
 | 
			
		||||
                                .title=_!workshop.title
 | 
			
		||||
                            %template.event-details{data: { href: view_workshop_path(@conference.slug, workshop.id) }}
 | 
			
		||||
                              %h1.title=_!workshop.title
 | 
			
		||||
                              %p.address
 | 
			
		||||
                                = _!("#{workshop.event_location.title}:")
 | 
			
		||||
                                = location_link workshop.event_location
 | 
			
		||||
                              .workshop-description= richtext workshop.info, 1
 | 
			
		||||
                            - if @can_edit
 | 
			
		||||
                              = form_tag administration_update_path(conference.slug, @admin_step), class: 'deschedule-workshop' do
 | 
			
		||||
                                .status
 | 
			
		||||
                                  .conflict-score
 | 
			
		||||
                                    %span.title Conflicts: 
 | 
			
		||||
                                    %span.value="#{status[:conflict_score]} / #{workshop.interested.size}"
 | 
			
		||||
                                = hidden_field_tag :id, workshop.id
 | 
			
		||||
                                = button :deschedule, value: :deschedule_workshop, class: [:delete, :small]
 | 
			
		||||
                        - elsif @can_edit
 | 
			
		||||
                          .title="Block #{time_data[:item][:block] + 1}"
 | 
			
		||||
                  - elsif time_data[:type] != :nil
 | 
			
		||||
                    %td{class: time_data[:type], rowspan: rowspan, colspan: data[:locations].present? ? data[:locations].size : 1}
 | 
			
		||||
                      - case time_data[:type]
 | 
			
		||||
                        - when :meal
 | 
			
		||||
                          - location = EventLocation.where(id: time_data[:item]['location'].to_i).first
 | 
			
		||||
                          - if location.present?
 | 
			
		||||
                            %a.event-detail-link
 | 
			
		||||
                              .details
 | 
			
		||||
                                .title=_!(time_data[:item]['title'])
 | 
			
		||||
                                .location=_!location.title
 | 
			
		||||
                            %template.event-details
 | 
			
		||||
                              %h1.title=_!(time_data[:item]['title'])
 | 
			
		||||
                              %p.address
 | 
			
		||||
                                = _!("#{location.title}:")
 | 
			
		||||
                                = location_link location
 | 
			
		||||
                        - when :event
 | 
			
		||||
                          - if time_data[:item].event_location.present?
 | 
			
		||||
                            %a.event-detail-link
 | 
			
		||||
                              .details
 | 
			
		||||
                                .title=_!(time_data[:item][:title]) if time_data[:item][:title]
 | 
			
		||||
                                .location=_!(time_data[:item].event_location.title)
 | 
			
		||||
                            %template.event-details
 | 
			
		||||
                              %h1.title=_!(time_data[:item][:title]) if time_data[:item][:title]
 | 
			
		||||
                              %p.address
 | 
			
		||||
                                = _!("#{time_data[:item].event_location.title}:")
 | 
			
		||||
                                = location_link time_data[:item].event_location
 | 
			
		||||
                              = richtext time_data[:item][:info], 1
 | 
			
		||||
                      %th=location.is_a?(Symbol) ? '' : _!(location.title)
 | 
			
		||||
              %tbody
 | 
			
		||||
                - data[:times].each do |time, time_data|
 | 
			
		||||
                  %tr{class: "row-type-#{time_data[:type] || 'nil'}"}
 | 
			
		||||
                    - rowspan = (time_data[:length] * (1 / @this_conference.schedule_interval)).to_i
 | 
			
		||||
                    %th=time(time).html_safe
 | 
			
		||||
                    - if time_data[:type] == :workshop
 | 
			
		||||
                      - data[:locations].each do |id, location|
 | 
			
		||||
                        - if time_data[:item][:workshops][id].present?
 | 
			
		||||
                          - workshop = time_data[:item][:workshops][id][:workshop]
 | 
			
		||||
                          - status = time_data[:item][:workshops][id][:status]
 | 
			
		||||
                        - else
 | 
			
		||||
                          - workshop = status = nil
 | 
			
		||||
                        %td{class: [time_data[:type], workshop.present? ? :filled : :open], rowspan: rowspan, data: workshop.present? ? nil : { block: time_data[:item][:block], day: day, location: id, division: division }}
 | 
			
		||||
                          - if workshop.present? && workshop.event_location.present?
 | 
			
		||||
                            .workshop-container
 | 
			
		||||
                              - if @can_edit
 | 
			
		||||
                                -if strip_tags(workshop.notes).strip.present?
 | 
			
		||||
                                  = admin_notes(workshop.notes)
 | 
			
		||||
                                - if status[:errors].present?
 | 
			
		||||
                                  = admin_status content_tag(:ul, (status[:errors].collect { |error| "<li>#{(_"errors.messages.schedule.#{error[:name].to_s}", vars: error[:i18nVars])}</li>"}.join).html_safe).html_safe
 | 
			
		||||
                              = link_to view_workshop_path(@conference.slug, workshop.id), class: 'event-detail-link' do
 | 
			
		||||
                                .details
 | 
			
		||||
                                  .title=_!workshop.title
 | 
			
		||||
                              %template.event-details{data: { href: view_workshop_path(@conference.slug, workshop.id) }}
 | 
			
		||||
                                %h1.title=_!workshop.title
 | 
			
		||||
                                %p.address
 | 
			
		||||
                                  = _!("#{workshop.event_location.title}:")
 | 
			
		||||
                                  = location_link workshop.event_location
 | 
			
		||||
                                .workshop-description= richtext workshop.info, 1
 | 
			
		||||
                              - if @can_edit
 | 
			
		||||
                                = form_tag administration_update_path(conference.slug, @admin_step), class: 'deschedule-workshop' do
 | 
			
		||||
                                  .status
 | 
			
		||||
                                    .conflict-score
 | 
			
		||||
                                      %span.title Conflicts: 
 | 
			
		||||
                                      %span.value="#{status[:conflict_score]} / #{workshop.interested.size}"
 | 
			
		||||
                                  = hidden_field_tag :id, workshop.id
 | 
			
		||||
                                  = button :deschedule, value: :deschedule_workshop, class: [:delete, :small]
 | 
			
		||||
                          - elsif @can_edit
 | 
			
		||||
                            .title="Block #{time_data[:item][:block] + 1}"
 | 
			
		||||
                    - elsif time_data[:type] != :nil
 | 
			
		||||
                      %td{class: time_data[:type], rowspan: rowspan, colspan: data[:locations].present? ? data[:locations].size : 1}
 | 
			
		||||
                        - case time_data[:type]
 | 
			
		||||
                          - when :meal
 | 
			
		||||
                            - location = EventLocation.where(id: time_data[:item]['location'].to_i).first
 | 
			
		||||
                            - if location.present?
 | 
			
		||||
                              %a.event-detail-link
 | 
			
		||||
                                .details
 | 
			
		||||
                                  .title=_!(time_data[:item]['title'])
 | 
			
		||||
                                  .location=_!location.title
 | 
			
		||||
                              %template.event-details
 | 
			
		||||
                                %h1.title=_!(time_data[:item]['title'])
 | 
			
		||||
                                %p.address
 | 
			
		||||
                                  = _!("#{location.title}:")
 | 
			
		||||
                                  = location_link location
 | 
			
		||||
                          - when :event
 | 
			
		||||
                            - if time_data[:item].event_location.present?
 | 
			
		||||
                              %a.event-detail-link
 | 
			
		||||
                                .details
 | 
			
		||||
                                  .title=_!(time_data[:item][:title]) if time_data[:item][:title]
 | 
			
		||||
                                  .location=_!(time_data[:item].event_location.title)
 | 
			
		||||
                              %template.event-details
 | 
			
		||||
                                %h1.title=_!(time_data[:item][:title]) if time_data[:item][:title]
 | 
			
		||||
                                %p.address
 | 
			
		||||
                                  = _!("#{time_data[:item].event_location.title}:")
 | 
			
		||||
                                  = location_link time_data[:item].event_location
 | 
			
		||||
                                = richtext time_data[:item][:info], 1
 | 
			
		||||
      - if @entire_page
 | 
			
		||||
        #workshop-selector
 | 
			
		||||
          = form_tag administration_update_path(@this_conference.slug, @admin_step), class: 'workshop-dlg', id: 'workshop-table-form' do
 | 
			
		||||
 | 
			
		||||
@ -5,7 +5,7 @@
 | 
			
		||||
			= @event_location.title
 | 
			
		||||
			= hidden_field_tag :event_location, @location
 | 
			
		||||
	- else
 | 
			
		||||
		= location_select(nil, inline_label: true, small: true, invalid_locations: (((((@schedule[@day] || {})[:times] || {})[@time] || {})[:item] || {})[:workshops] || {}).keys, label: false)
 | 
			
		||||
		= location_select(nil, inline_label: true, small: true, invalid_locations: @invalid_locations, label: false)
 | 
			
		||||
- if @event_location.present?
 | 
			
		||||
	.host-field
 | 
			
		||||
		%h4.inline=_'articles.admin.locations.headings.amenities'
 | 
			
		||||
@ -38,7 +38,7 @@
 | 
			
		||||
			%td=(workshop.active_facilitators.map { |x| x.named_email }).join(', ')
 | 
			
		||||
			%td=workshop.interested_count
 | 
			
		||||
			%td
 | 
			
		||||
				.text=workshop.notes
 | 
			
		||||
				.text=strip_tags(workshop.notes)
 | 
			
		||||
 | 
			
		||||
.legend
 | 
			
		||||
	%h4 Legend
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										12
									
								
								app/views/conference_administration/_workshops.html.haml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								app/views/conference_administration/_workshops.html.haml
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,12 @@
 | 
			
		||||
- add_inline_script :registrations
 | 
			
		||||
= columns(medium: 12) do
 | 
			
		||||
  .goes-fullscreen#registrations-table
 | 
			
		||||
    .flex-column
 | 
			
		||||
      = searchfield :search, nil, big: true, stretch: true
 | 
			
		||||
      %a.button{data: { expands: 'registrations-table' }}='expand'
 | 
			
		||||
      %a.button.delete{data: { contracts: 'registrations-table' }}='close'
 | 
			
		||||
    .table-scroller#registrations
 | 
			
		||||
      = html_table(@excel_data, workshops_table_options)
 | 
			
		||||
= columns(medium: 12) do
 | 
			
		||||
  .actions.center
 | 
			
		||||
    = link_to (_'links.download.Excel'), administration_step_path(@this_conference.slug, :workshops, format: :xlsx), class: [:button, :download]
 | 
			
		||||
@ -1427,6 +1427,7 @@ en:
 | 
			
		||||
          locations: Locations
 | 
			
		||||
          meals: Meals
 | 
			
		||||
          events: Events
 | 
			
		||||
          workshops: Workshops
 | 
			
		||||
        descriptions:
 | 
			
		||||
          locations: Create the list of locations that you will be using for events,
 | 
			
		||||
            meals, and workshops.
 | 
			
		||||
@ -1435,6 +1436,7 @@ en:
 | 
			
		||||
          events: Create event details. These events should be any type of event other
 | 
			
		||||
            than meals and workshops such as meeting, rides, and parties. The events
 | 
			
		||||
            will be added to your schedule.
 | 
			
		||||
          workshops: View and modify all workshops created by registered users.
 | 
			
		||||
      schedule:
 | 
			
		||||
        description: On this page you can schedule workshops and publish your schedule
 | 
			
		||||
          to the front page.
 | 
			
		||||
@ -2625,6 +2627,7 @@ en:
 | 
			
		||||
        unregistered: Unregistered
 | 
			
		||||
        facilitator: Facilitator
 | 
			
		||||
  workshop:
 | 
			
		||||
    created_at: Creation Date
 | 
			
		||||
    options:
 | 
			
		||||
      needs:
 | 
			
		||||
        projector: Projector
 | 
			
		||||
@ -2718,6 +2721,7 @@ en:
 | 
			
		||||
    content:
 | 
			
		||||
      change_locale: Read in %{language}
 | 
			
		||||
      Translation_of: Translation of
 | 
			
		||||
      item_translation: "%{item} in %{language}"
 | 
			
		||||
  number:
 | 
			
		||||
    currency:
 | 
			
		||||
      format:
 | 
			
		||||
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user