Merge remote-tracking branch 'upstream/master'

This commit is contained in:
Dan Rice 2013-03-19 16:47:20 -04:00
commit 59989d3127
9 changed files with 211 additions and 200 deletions

View file

@ -178,18 +178,18 @@ class StatsController < ApplicationController
end
def context_total_actions_data
all_actions_per_context = Stats::TopContextsQuery.new(current_user).result
prep_context_data_for_view(all_actions_per_context)
actions = Stats::TopContextsQuery.new(current_user).result
render :layout => false
@data = Stats::PieChartData.new(actions, t('stats.spread_of_actions_for_all_context'), 70)
render :pie_chart_data, :layout => false
end
def context_running_actions_data
all_actions_per_context = Stats::TopContextsQuery.new(current_user, :running => true).result
actions = Stats::TopContextsQuery.new(current_user, :running => true).result
@data = Stats::PieChartData.new(actions, t('stats.spread_of_running_actions_for_visible_contexts'), 60)
prep_context_data_for_view(all_actions_per_context)
render :layout => false
render :pie_chart_data, :layout => false
end
def actions_day_of_week_all_data
@ -341,30 +341,6 @@ class StatsController < ApplicationController
private
def prep_context_data_for_view(all_actions_per_context)
@sum = all_actions_per_context.inject(0){|sum, apc| sum += apc['total'].to_i }
pie_cutoff=10
size = [all_actions_per_context.size, pie_cutoff].min
# explicitely copy contents of hash to avoid ending up with two arrays pointing to same hashes
@actions_per_context = Array.new(size){|i| {
'name' => all_actions_per_context[i]['name'],
'total' => all_actions_per_context[i]['total'].to_i,
'id' => all_actions_per_context[i]['id']
} }
if all_actions_per_context.size > pie_cutoff
@actions_per_context[size-1]['name']=t('stats.other_actions_label')
@actions_per_context[size-1]['total']=@actions_per_context[size-1]['total']
@actions_per_context[size-1]['id']=-1
size.upto(all_actions_per_context.size-1){ |i| @actions_per_context[size-1]['total']+=(all_actions_per_context[i]['total'].to_i) }
end
@truncate_chars = 15
end
def init
@me = self # for meta programming

View file

@ -0,0 +1,86 @@
module Stats
class PieChartData
attr_reader :all_totals, :alpha, :title
def initialize(all_totals, title, alpha)
@all_totals = all_totals
@title = title
@alpha = alpha
end
def values
@values ||= Array.new(slices) do |i|
chart_totals[i]['total'] * 100 / sum
end
end
def labels
@labels ||= Array.new(slices) do |i|
chart_totals[i]['name'].truncate(15, :omission => '...')
end
end
def ids
@ids ||= Array.new(slices) do |i|
chart_totals[i]['id']
end
end
def sum
@sum ||= totals.inject(0) do |sum, total|
sum + total
end
end
private
def pie_cutoff
10
end
def slices
@slices ||= [all_totals.size, pie_cutoff].min
end
def subtotal(from, to)
totals[from..to].inject(0) do |sum, total|
sum + total
end
end
def chart_totals
unless @chart_totals
@chart_totals = first_n_totals(10)
if all_totals.size > pie_cutoff
@chart_totals[-1] = other
end
end
@chart_totals
end
def first_n_totals(n)
# create a duplicate so that we don't accidentally
# overwrite the original array
Array.new(slices) do |i|
{
'name' => all_totals[i]['name'],
'total' => all_totals[i]['total'],
'id' => all_totals[i]['id']
}
end
end
def other
{
'name' => I18n.t('stats.other_actions_label'),
'id' => -1,
'total' => subtotal(slices-1, all_totals.size-1)
}
end
def totals
@totals ||= all_totals.map { |item| item['total'] }
end
end
end

View file

@ -1,15 +0,0 @@
<%
size = @actions_per_context.size()
pie_slices = Array.new(size){|i| @actions_per_context[i]['total'].to_i*100/@sum }
pie_labels = Array.new(size){|i| truncate(@actions_per_context[i]['name'], :length => @truncate_chars, :omission => '...') }
pie_links = Array.new(size){|i| url_for :controller => "contexts", :action => "show", :id=>@actions_per_context[i]['id']}
-%>
&title=<%= t('stats.spread_of_running_actions_for_visible_contexts') %>,{font-size:16}&
&pie=60,#505050,{font-size: 12px; color: #404040;}&
&x_axis_steps=1& &y_ticks=5,10,5& &line=3,#87421F& &y_min=0& &y_max=20&
&values=<%= pie_slices.join(",") %>&
&pie_labels=<%= pie_labels.join(",") %>&
&links=<%= pie_links.join(",") %>&
&colours=#d01f3c,#356aa0,#C79810,#c61fd0,#1fc6d0,#1fd076,#72d01f,#c6d01f,#d0941f,#40941f&
&tool_tip=#x_label#: #val#%25&
&x_label_style=9,,2,1&

View file

@ -1,15 +0,0 @@
<%
size = @actions_per_context.size()
pie_slices = Array.new(size){|i| @actions_per_context[i]['total'].to_i*100/@sum }
pie_labels = Array.new(size){|i| truncate(@actions_per_context[i]['name'], :length => @truncate_chars, :omission => '...') }
pie_links = Array.new(size){|i| url_for :controller => "contexts", :action => "show", :id=>@actions_per_context[i]['id']}
-%>
&title=<%= t('stats.spread_of_actions_for_all_context') %>,{font-size:16}&
&pie=70,#505050,{font-size: 12px; color: #404040;}&
&x_axis_steps=1& &y_ticks=5,10,5& &line=3,#87421F& &y_min=0& &y_max=20&
&values=<%= pie_slices.join(",") %>&
&pie_labels=<%= pie_labels.join(",") %>&
&links=<%= pie_links.join(",") %>&
&colours=#d01f3c,#356aa0,#C79810,#c61fd0,#1fc6d0,#1fd076,#72d01f,#c6d01f,#d0941f,#40941f&
&tool_tip=#x_label#: #val#%25&
&x_label_style=9,,2,1&

View file

@ -0,0 +1,9 @@
&title=<%= @data.title %>,{font-size:16}&
&pie=<%= @data.alpha %>,#505050,{font-size: 12px; color: #404040;}&
&x_axis_steps=1& &y_ticks=5,10,5& &line=3,#87421F& &y_min=0& &y_max=20&
&values=<%= @data.values.join(",") %>&
&pie_labels=<%= @data.labels.join(",") %>&
&links=<%= @data.ids.map{|id| context_path(id)}.join(",") %>&
&colours=#d01f3c,#356aa0,#C79810,#c61fd0,#1fc6d0,#1fd076,#72d01f,#c6d01f,#d0941f,#40941f&
&tool_tip=#x_label#: #val#%25&
&x_label_style=9,,2,1&

View file

@ -35,7 +35,7 @@ Feature: Show done
Scenario Outline: I can see all todos completed in the last timeperiod
When I go to the <page>
Then I should see "todo 1"
And I should see "Completed Today"
And I should see "Completed today"
And I should see "Completed in the rest of this week"
And I should see "Completed in the rest of this month"

View file

@ -3,152 +3,32 @@ require File.expand_path(File.dirname(__FILE__) + '/../test_helper')
class ContextActionsDataTest < ActionController::TestCase
tests StatsController
def test_total_with_0_items
login_as(:admin_user)
Stats::TopContextsQuery.any_instance.stubs(:result).returns []
get :context_total_actions_data
assert_equal [], assigns[:actions_per_context]
end
def test_total_with_less_than_10_items
login_as(:admin_user)
contexts = [
{'id' => 1, 'name' => 'one', 'total' => 11},
{'id' => 2, 'name' => 'two', 'total' => 4},
{'id' => 3, 'name' => 'three', 'total' => 8},
{'id' => 4, 'name' => 'four', 'total' => 13},
{'id' => 5, 'name' => 'five', 'total' => 20},
{'id' => 6, 'name' => 'six', 'total' => 17},
{'id' => 7, 'name' => 'seven', 'total' => 5},
{'id' => 8, 'name' => 'eight', 'total' => 1},
{'id' => 9, 'name' => 'nine', 'total' => 6}
]
Stats::TopContextsQuery.any_instance.stubs(:result).returns contexts
get :context_total_actions_data
assert_equal contexts, assigns[:actions_per_context]
end
def test_total_with_exactly_10_items
login_as(:admin_user)
contexts = [
{'id' => 1, 'name' => 'one', 'total' => 11},
{'id' => 2, 'name' => 'two', 'total' => 4},
{'id' => 3, 'name' => 'three', 'total' => 8},
{'id' => 4, 'name' => 'four', 'total' => 13},
{'id' => 5, 'name' => 'five', 'total' => 20},
{'id' => 6, 'name' => 'six', 'total' => 17},
{'id' => 7, 'name' => 'seven', 'total' => 5},
{'id' => 8, 'name' => 'eight', 'total' => 1},
{'id' => 9, 'name' => 'nine', 'total' => 6},
{'id' => 10, 'name' => 'ten', 'total' => 19}
]
Stats::TopContextsQuery.any_instance.stubs(:result).returns contexts
get :context_total_actions_data
assert_equal contexts, assigns[:actions_per_context]
end
def test_total_with_more_than_10_items
login_as(:admin_user)
contexts = [
{'id' => 1, 'name' => 'one', 'total' => 11},
{'id' => 2, 'name' => 'two', 'total' => 4},
{'id' => 3, 'name' => 'three', 'total' => 8},
{'id' => 4, 'name' => 'four', 'total' => 13},
{'id' => 5, 'name' => 'five', 'total' => 20},
{'id' => 6, 'name' => 'six', 'total' => 17},
{'id' => 7, 'name' => 'seven', 'total' => 5},
{'id' => 8, 'name' => 'eight', 'total' => 1},
{'id' => 9, 'name' => 'nine', 'total' => 6},
{'id' => 10, 'name' => 'ten', 'total' => 19},
{'id' => 11, 'name' => 'eleven', 'total' => 14}
{'id' => 3, 'name' => 'three', 'total' => 8}
]
Stats::TopContextsQuery.any_instance.stubs(:result).returns contexts
get :context_total_actions_data
contexts.pop
contexts[-1] = {'id' => -1, 'name' => '(others)', 'total' => 33}
assert_equal contexts, assigns[:actions_per_context]
assert_equal [47, 17, 34], assigns[:data].values
end
def test_running_with_0_items
login_as(:admin_user)
Stats::TopContextsQuery.any_instance.stubs(:result).returns []
get :context_running_actions_data
assert_equal [], assigns[:actions_per_context]
end
def test_running_with_less_than_10_items
def test_running_actions
login_as(:admin_user)
contexts = [
{'id' => 1, 'name' => 'one', 'total' => 11},
{'id' => 2, 'name' => 'two', 'total' => 4},
{'id' => 3, 'name' => 'three', 'total' => 8},
{'id' => 4, 'name' => 'four', 'total' => 13},
{'id' => 5, 'name' => 'five', 'total' => 20},
{'id' => 6, 'name' => 'six', 'total' => 17},
{'id' => 7, 'name' => 'seven', 'total' => 5},
{'id' => 8, 'name' => 'eight', 'total' => 1},
{'id' => 9, 'name' => 'nine', 'total' => 6}
{'id' => 3, 'name' => 'three', 'total' => 8}
]
Stats::TopContextsQuery.any_instance.stubs(:result).returns contexts
get :context_running_actions_data
assert_equal contexts, assigns[:actions_per_context]
end
def test_running_with_exactly_10_items
login_as(:admin_user)
contexts = [
{'id' => 1, 'name' => 'one', 'total' => 11},
{'id' => 2, 'name' => 'two', 'total' => 4},
{'id' => 3, 'name' => 'three', 'total' => 8},
{'id' => 4, 'name' => 'four', 'total' => 13},
{'id' => 5, 'name' => 'five', 'total' => 20},
{'id' => 6, 'name' => 'six', 'total' => 17},
{'id' => 7, 'name' => 'seven', 'total' => 5},
{'id' => 8, 'name' => 'eight', 'total' => 1},
{'id' => 9, 'name' => 'nine', 'total' => 6},
{'id' => 10, 'name' => 'ten', 'total' => 19}
]
Stats::TopContextsQuery.any_instance.stubs(:result).returns contexts
get :context_running_actions_data
assert_equal contexts, assigns[:actions_per_context]
end
def test_running_with_more_than_10_items
login_as(:admin_user)
contexts = [
{'id' => 1, 'name' => 'one', 'total' => 11},
{'id' => 2, 'name' => 'two', 'total' => 4},
{'id' => 3, 'name' => 'three', 'total' => 8},
{'id' => 4, 'name' => 'four', 'total' => 13},
{'id' => 5, 'name' => 'five', 'total' => 20},
{'id' => 6, 'name' => 'six', 'total' => 17},
{'id' => 7, 'name' => 'seven', 'total' => 5},
{'id' => 8, 'name' => 'eight', 'total' => 1},
{'id' => 9, 'name' => 'nine', 'total' => 6},
{'id' => 10, 'name' => 'ten', 'total' => 19},
{'id' => 11, 'name' => 'eleven', 'total' => 14}
]
Stats::TopContextsQuery.any_instance.stubs(:result).returns contexts
get :context_running_actions_data
contexts.pop
contexts[-1] = {'id' => -1, 'name' => '(others)', 'total' => 33}
assert_equal contexts, assigns[:actions_per_context]
assert_equal [47, 17, 34], assigns[:data].values
end
end

View file

@ -26,12 +26,19 @@ class StatsControllerTest < ActionController::TestCase
actions_day_of_week_30days_data
actions_time_of_day_all_data
actions_time_of_day_30days_data
}.each do |action|
get action
assert_response :success
assert_template "stats/"+action
end
%w{
context_total_actions_data
context_running_actions_data
}.each do |action|
get action
assert_response :success
assert_template "stats/"+action
assert_template "stats/pie_chart_data"
end
end
@ -262,8 +269,8 @@ class StatsControllerTest < ActionController::TestCase
get :context_total_actions_data
assert_response :success
assert_equal 9, assigns['sum'], "Nine todos in 1 context"
assert_equal 1, assigns['actions_per_context'].size
assert_equal 9, assigns['data'].sum, "Nine todos in 1 context"
assert_equal 1, assigns['data'].values.size
# Given 10 more todos in 10 different contexts
1.upto(10) do |i|
@ -275,10 +282,10 @@ class StatsControllerTest < ActionController::TestCase
get :context_total_actions_data
assert_response :success
assert_equal 19, assigns['sum'], "added 10 todos"
assert_equal 10, assigns['actions_per_context'].size, "pie slices limited to max 10"
assert_equal 2, assigns['actions_per_context'][9]['total'], "pie slices limited to max 10; last pie contains sum of rest"
assert_equal "(others)", assigns['actions_per_context'][9]['name'], "pie slices limited to max 10; last slice contains label for others"
assert_equal 19, assigns['data'].sum, "added 10 todos"
assert_equal 10, assigns['data'].values.size, "pie slices limited to max 10"
assert_equal 10, assigns['data'].values[9], "pie slices limited to max 10; last pie contains sum of rest (in percentage)"
assert_equal "(others)", assigns['data'].labels[9], "pie slices limited to max 10; last slice contains label for others"
end
def test_context_running_actions_data
@ -292,8 +299,8 @@ class StatsControllerTest < ActionController::TestCase
get :context_running_actions_data
assert_response :success
assert_equal 4, assigns['sum'], "Four running todos in 1 context"
assert_equal 1, assigns['actions_per_context'].size
assert_equal 4, assigns['data'].sum, "Four todos in 1 context"
assert_equal 1, assigns['data'].values.size
# Given 10 more todos in 10 different contexts
1.upto(10) do |i|
@ -305,10 +312,9 @@ class StatsControllerTest < ActionController::TestCase
get :context_running_actions_data
assert_response :success
assert_equal 14, assigns['sum'], "added 10 todos"
assert_equal 10, assigns['actions_per_context'].size, "pie slices limited to max 10"
assert_equal 2, assigns['actions_per_context'][9]['total'], "pie slices limited to max 10; last pie contains sum of rest"
assert_equal "(others)", assigns['actions_per_context'][9]['name'], "pie slices limited to max 10; last slice contains label for others"
assert_equal 10, assigns['data'].values.size, "pie slices limited to max 10"
assert_equal 14, assigns['data'].values[9], "pie slices limited to max 10; last pie contains sum of rest (in percentage)"
assert_equal "(others)", assigns['data'].labels[9], "pie slices limited to max 10; last slice contains label for others"
end
def test_actions_day_of_week_all_data

View file

@ -0,0 +1,84 @@
require File.expand_path(File.dirname(__FILE__) + '/../minimal_test_helper')
require 'app/models/stats/pie_chart_data'
require 'active_support/core_ext/string'
class Stats::PieChartDataTest < Test::Unit::TestCase
def setup
xx = { :stats => { :other_actions_label => '(other)' } }
I18n.backend.store_translations(:xx, xx)
I18n.locale = :xx
end
def test_with_0_items
data = Stats::PieChartData.new([], 'a chart', 50)
assert_equal [], data.values
assert_equal [], data.labels
assert_equal [], data.ids
end
def test_with_less_than_10_items
items = [
{'id' => 1, 'name' => 'one', 'total' => 11},
{'id' => 2, 'name' => 'two', 'total' => 4},
{'id' => 3, 'name' => 'three', 'total' => 8},
{'id' => 4, 'name' => 'four', 'total' => 13},
{'id' => 5, 'name' => 'five', 'total' => 20},
{'id' => 6, 'name' => 'six', 'total' => 17},
{'id' => 7, 'name' => 'seven', 'total' => 5},
{'id' => 8, 'name' => 'eight', 'total' => 1},
{'id' => 9, 'name' => 'nine', 'total' => 6}
]
data = Stats::PieChartData.new(items, 'a chart', 50)
assert_equal [12, 4, 9, 15, 23, 20, 5, 1, 7], data.values
assert_equal ["one", "two", "three", "four", "five", "six", "seven", "eight", "nine"], data.labels
assert_equal [1, 2, 3, 4, 5, 6, 7, 8, 9], data.ids
end
def test_with_exactly_10_items
items = [
{'id' => 1, 'name' => 'one', 'total' => 11},
{'id' => 2, 'name' => 'two', 'total' => 4},
{'id' => 3, 'name' => 'three', 'total' => 8},
{'id' => 4, 'name' => 'four', 'total' => 13},
{'id' => 5, 'name' => 'five', 'total' => 20},
{'id' => 6, 'name' => 'six', 'total' => 17},
{'id' => 7, 'name' => 'seven', 'total' => 5},
{'id' => 8, 'name' => 'eight', 'total' => 1},
{'id' => 9, 'name' => 'nine', 'total' => 6},
{'id' => 10, 'name' => 'ten', 'total' => 19}
]
data = Stats::PieChartData.new(items, 'a chart', 50)
assert_equal [10, 3, 7, 12, 19, 16, 4, 0, 5, 18], data.values
assert_equal ["one", "two", "three", "four", "five", "six", "seven", "eight", "nine", "ten"], data.labels
assert_equal [1, 2, 3, 4, 5, 6, 7, 8, 9, 10], data.ids
end
def test_with_more_than_10_items
items = [
{'id' => 1, 'name' => 'one', 'total' => 11},
{'id' => 2, 'name' => 'two', 'total' => 4},
{'id' => 3, 'name' => 'three', 'total' => 8},
{'id' => 4, 'name' => 'four', 'total' => 13},
{'id' => 5, 'name' => 'five', 'total' => 20},
{'id' => 6, 'name' => 'six', 'total' => 17},
{'id' => 7, 'name' => 'seven', 'total' => 5},
{'id' => 8, 'name' => 'eight', 'total' => 1},
{'id' => 9, 'name' => 'nine', 'total' => 6},
{'id' => 10, 'name' => 'ten', 'total' => 19},
{'id' => 11, 'name' => 'eleven', 'total' => 14}
]
data = Stats::PieChartData.new(items, 'a chart', 50)
assert_equal [9, 3, 6, 11, 16, 14, 4, 0, 5, 27], data.values
assert_equal ["one", "two", "three", "four", "five", "six", "seven", "eight", "nine", "(other)"], data.labels
assert_equal [1, 2, 3, 4, 5, 6, 7, 8, 9, -1], data.ids
end
end