/*
* Copyright 2013 Canonical Ltd.
* Copyright 2021 Robert Tari
*
* This program is free software: you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 3, as published
* by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranties of
* MERCHANTABILITY, SATISFACTORY QUALITY, or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program. If not, see .
*
* Authors:
* Charles Kerr
* Robert Tari
*/
#include "state-mock.h"
#include "timedated-fixture.h"
#include
extern "C"
{
#include
}
using namespace ayatana::indicator::datetime;
class MockLiveActions: public LiveActions
{
public:
explicit MockLiveActions(const std::shared_ptr& state_in): LiveActions(state_in) {}
~MockLiveActions() {}
};
class TestLiveActionsFixture: public TimedatedFixture
{
private:
using super = TimedatedFixture;
protected:
std::shared_ptr m_mock_state;
std::shared_ptr m_state;
std::shared_ptr m_live_actions;
std::shared_ptr m_actions;
void SetUp() override
{
super::SetUp();
// create the State and Actions
m_mock_state.reset(new MockState);
m_mock_state->settings.reset(new Settings);
m_state = std::dynamic_pointer_cast(m_mock_state);
m_live_actions.reset(new MockLiveActions(m_state));
m_actions = std::dynamic_pointer_cast(m_live_actions);
// start the timedate1 dbusmock
start_timedate1("Etc/Utc");
}
void TearDown() override
{
m_actions.reset();
m_live_actions.reset();
m_state.reset();
m_mock_state.reset();
super::TearDown();
}
};
/***
****
***/
TEST_F(TestLiveActionsFixture, HelloWorld)
{
EXPECT_TRUE(true);
}
TEST_F(TestLiveActionsFixture, SetLocation)
{
const std::string tzid = "America/Chicago";
const std::string name = "Oklahoma City";
const std::string expected = tzid + " " + name;
EXPECT_NE(expected, m_state->settings->timezone_name.get());
std::string new_name;
m_state->settings->timezone_name.changed().connect(
[&new_name](const std::string& n){new_name = n;}
);
m_actions->set_location(tzid, name);
EXPECT_TRUE(wait_for([&new_name](){return !new_name.empty();}));
EXPECT_EQ(expected, new_name);
EXPECT_EQ(expected, m_state->settings->timezone_name.get());
EXPECT_EQ(tzid, get_timedate1_timezone());
}
/***
****
***/
TEST_F(TestLiveActionsFixture, DesktopOpenAlarmApp)
{
std::string sReturn = m_actions->open_alarm_app();
std::string expected = "evolution -c calendar";
if (ayatana_common_utils_is_lomiri())
{
expected = "alarm://";
}
EXPECT_EQ(expected, sReturn);
}
TEST_F(TestLiveActionsFixture, DesktopOpenAppointment)
{
if (ayatana_common_utils_is_lomiri())
{
// PhoneOpenAppointment will handle this
return;
}
Appointment a;
a.uid = "some-uid";
a.begin = DateTime::NowLocal();
std::string sReturn = m_actions->open_appointment(a, a.begin);
const std::string expected_substr = "evolution \"calendar:///?startdate=";
EXPECT_NE(sReturn.find(expected_substr), std::string::npos);
}
TEST_F(TestLiveActionsFixture, DesktopOpenCalendarApp)
{
if (ayatana_common_utils_is_lomiri())
{
// PhoneOpenCalendarApp will handle this
return;
}
std::string sReturn = m_actions->open_calendar_app(DateTime::NowLocal());
const std::string expected_substr = "evolution \"calendar:///?startdate=";
EXPECT_NE(sReturn.find(expected_substr), std::string::npos);
}
TEST_F(TestLiveActionsFixture, DesktopOpenSettingsApp)
{
std::string sReturn = m_actions->open_settings_app();
std::string expected_substr = "gnome-control-center datetime";
if (ayatana_common_utils_is_unity())
{
expected_substr = "unity-control-center datetime";
}
else if (ayatana_common_utils_is_mate())
{
expected_substr = "mate-time-admin";
}
else if (ayatana_common_utils_is_lomiri())
{
expected_substr = "settings:///system/time-date";
}
EXPECT_EQ(expected_substr, sReturn);
}
/***
****
***/
namespace
{
const std::string clock_app_url = "alarm://";
}
TEST_F(TestLiveActionsFixture, PhoneOpenAlarmApp)
{
setenv("XDG_CURRENT_DESKTOP", "Lomiri", 1);
std::string sReturn = m_actions->open_alarm_app();
EXPECT_EQ(clock_app_url, sReturn);
unsetenv("XDG_CURRENT_DESKTOP");
}
TEST_F(TestLiveActionsFixture, PhoneOpenAppointment)
{
setenv("XDG_CURRENT_DESKTOP", "Lomiri", 1);
Appointment a;
a.uid = "event-uid";
a.source_uid = "source-uid";
a.begin = DateTime::NowLocal();
a.type = Appointment::EVENT;
auto ocurrenceDate = DateTime::Local(2014, 1, 1, 0, 0, 0);
std::string sReturn = m_actions->open_appointment(a, ocurrenceDate);
const std::string appointment_app_url = ocurrenceDate.to_timezone("UTC").format("calendar://startdate=%Y-%m-%dT%H:%M:%S+00:00");
EXPECT_EQ(appointment_app_url, sReturn);
a.type = Appointment::ALARM;
sReturn = m_actions->open_appointment(a, a.begin);
EXPECT_EQ(clock_app_url, sReturn);
unsetenv("XDG_CURRENT_DESKTOP");
}
TEST_F(TestLiveActionsFixture, PhoneOpenCalendarApp)
{
setenv("XDG_CURRENT_DESKTOP", "Lomiri", 1);
auto now = DateTime::NowLocal();
std::string sReturn = m_actions->open_calendar_app(now);
const std::string expected = now.to_timezone("UTC").format("calendar://startdate=%Y-%m-%dT%H:%M:%S+00:00");
EXPECT_EQ(expected, sReturn);
unsetenv("XDG_CURRENT_DESKTOP");
}
TEST_F(TestLiveActionsFixture, PhoneOpenSettingsApp)
{
setenv("XDG_CURRENT_DESKTOP", "Lomiri", 1);
std::string sReturn = m_actions->open_settings_app();
const std::string expected = "settings:///system/time-date";
EXPECT_EQ(expected, sReturn);
unsetenv("XDG_CURRENT_DESKTOP");
}
/***
****
***/
TEST_F(TestLiveActionsFixture, CalendarState)
{
// init the clock
auto now = DateTime::Local(2014, 1, 1, 0, 0, 0);
m_mock_state->mock_clock->set_localtime (now);
m_state->calendar_month->month().set(now);
//m_state->planner->time.set(now);
///
/// Test the default calendar state.
///
auto action_group = m_actions->action_group();
auto calendar_state = g_action_group_get_action_state (action_group, "calendar");
EXPECT_TRUE (calendar_state != nullptr);
EXPECT_TRUE (g_variant_is_of_type (calendar_state, G_VARIANT_TYPE_DICTIONARY));
// there's nothing in the planner yet, so appointment-days should be an empty array
auto v = g_variant_lookup_value (calendar_state, "appointment-days", G_VARIANT_TYPE_ARRAY);
EXPECT_TRUE (v != nullptr);
EXPECT_EQ (0, g_variant_n_children (v));
g_clear_pointer (&v, g_variant_unref);
// calendar-day should be in sync with m_state->calendar_day
v = g_variant_lookup_value (calendar_state, "calendar-day", G_VARIANT_TYPE_INT64);
EXPECT_TRUE (v != nullptr);
EXPECT_EQ (m_state->calendar_month->month().get().to_unix(), g_variant_get_int64(v));
g_clear_pointer (&v, g_variant_unref);
// show-week-numbers should be false because MockSettings defaults everything to 0
v = g_variant_lookup_value (calendar_state, "show-week-numbers", G_VARIANT_TYPE_BOOLEAN);
EXPECT_TRUE (v != nullptr);
EXPECT_FALSE (g_variant_get_boolean (v));
g_clear_pointer (&v, g_variant_unref);
// cleanup this step
g_clear_pointer (&calendar_state, g_variant_unref);
///
/// Now add appointments to the planner and confirm that the state keeps in sync
///
auto tomorrow = now.add_days(1);
auto tomorrow_begin = tomorrow.start_of_day();
auto tomorrow_end = tomorrow.end_of_day();
Appointment a1;
a1.color = "green";
a1.summary = "write unit tests";
a1.uid = "D4B57D50247291478ED31DED17FF0A9838DED402";
a1.begin = tomorrow_begin;
a1.end = tomorrow_end;
auto ubermorgen = now.add_days(2);
auto ubermorgen_begin = ubermorgen.start_of_day();
auto ubermorgen_end = ubermorgen.end_of_day();
Appointment a2;
a2.color = "orange";
a2.summary = "code review";
a2.uid = "2756ff7de3745bbffd65d2e4779c37c7ca60d843";
a2.begin = ubermorgen_begin;
a2.end = ubermorgen_end;
m_state->calendar_month->appointments().set(std::vector({a1, a2}));
///
/// Now test the calendar state again.
/// The this_month field should now contain the appointments we just added.
///
calendar_state = g_action_group_get_action_state (action_group, "calendar");
v = g_variant_lookup_value (calendar_state, "appointment-days", G_VARIANT_TYPE_ARRAY);
EXPECT_TRUE (v != nullptr);
int i;
g_variant_get_child (v, 0, "i", &i);
EXPECT_EQ (a1.begin.day_of_month(), i);
g_variant_get_child (v, 1, "i", &i);
EXPECT_EQ (a2.begin.day_of_month(), i);
g_clear_pointer(&v, g_variant_unref);
g_clear_pointer(&calendar_state, g_variant_unref);
///
/// Confirm that the action state's dictionary
/// keeps in sync with settings.show_week_numbers
///
auto b = m_state->settings->show_week_numbers.get();
for (i=0; i<2; i++)
{
b = !b;
m_state->settings->show_week_numbers.set(b);
calendar_state = g_action_group_get_action_state (action_group, "calendar");
v = g_variant_lookup_value (calendar_state, "show-week-numbers", G_VARIANT_TYPE_BOOLEAN);
EXPECT_TRUE(v != nullptr);
EXPECT_EQ(b, g_variant_get_boolean(v));
g_clear_pointer(&v, g_variant_unref);
g_clear_pointer(&calendar_state, g_variant_unref);
}
}