openlp/tests/openlp_plugins/bibles/test_mediaitem.py

1688 lines
75 KiB
Python

# -*- coding: utf-8 -*-
##########################################################################
# OpenLP - Open Source Lyrics Projection #
# ---------------------------------------------------------------------- #
# Copyright (c) 2008-2022 OpenLP Developers #
# ---------------------------------------------------------------------- #
# This program is free software: you can redistribute it and/or modify #
# it under the terms of the GNU General Public License as published by #
# the Free Software Foundation, either version 3 of the License, or #
# (at your option) any later version. #
# #
# This program is distributed in the hope that it will be useful, #
# but WITHOUT ANY WARRANTY; without even the implied warranty of #
# MERCHANTABILITY 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 <https://www.gnu.org/licenses/>. #
##########################################################################
"""
This module contains tests for the lib submodule of the Presentations plugin.
"""
import pytest
from unittest.mock import MagicMock, call, patch
from PyQt5 import QtCore, QtWidgets
from openlp.core.common.registry import Registry
from openlp.core.lib.mediamanageritem import MediaManagerItem
from openlp.plugins.bibles.lib.mediaitem import VALID_TEXT_SEARCH, BibleMediaItem, BibleSearch, ResultsTab, \
SearchStatus, SearchTabs, get_reference_separators
@pytest.fixture()
def media_item(mock_settings):
Registry().register('main_window', MagicMock())
with patch('openlp.plugins.bibles.lib.mediaitem.MediaManagerItem._setup'), \
patch('openlp.plugins.bibles.lib.mediaitem.BibleMediaItem.setup_item'):
m_item = BibleMediaItem(None, MagicMock())
media_item.plugin = MagicMock()
m_item.results_view_tab = MagicMock()
return m_item
mocked_book_1 = MagicMock(**{'get_name.return_value': 'Book 1', 'book_reference_id': 1})
mocked_book_2 = MagicMock(**{'get_name.return_value': 'Book 2', 'book_reference_id': 2})
mocked_book_3 = MagicMock(**{'get_name.return_value': 'Book 3', 'book_reference_id': 3})
mocked_book_4 = MagicMock(**{'get_name.return_value': 'Book 4', 'book_reference_id': 4})
book_list_1 = [mocked_book_1, mocked_book_2, mocked_book_3]
book_list_2 = [mocked_book_2, mocked_book_3, mocked_book_4]
mocked_bible_1 = MagicMock(**{'get_books.return_value': book_list_1})
mocked_bible_1.name = 'Bible 1'
mocked_bible_2 = MagicMock(**{'get_books.return_value': book_list_2})
mocked_bible_2.name = 'Bible 2'
@pytest.fixture()
def mocked_timer():
return patch('openlp.plugins.bibles.lib.mediaitem.QtCore.QTimer').start()
@pytest.fixture()
def mocked_log():
return patch('openlp.plugins.bibles.lib.mediaitem.log').start()
def test_valid_text_search():
"""
Test the compiled VALID_TEXT_SEARCH regex expression
"""
# GIVEN: Some test data and some expected results
test_data = [('a a a', None), ('a ab a', None), ('a abc a', ((2, 5),)), ('aa 123 aa', ((3, 6),))]
for data, expected_result in test_data:
# WHEN: Calling search on the compiled regex expression
result = VALID_TEXT_SEARCH.search(data)
# THEN: The expected result should be returned
if expected_result is None:
assert result is None, expected_result
else:
assert result.regs == expected_result
def test_get_reference_separators():
"""
Test the module function get_reference_separators
"""
# GIVEN: A mocked get_reference_separator from the :mod:`openlp.plugins.bibles.lib` module
with patch('openlp.plugins.bibles.lib.mediaitem.get_reference_separator') as mocked_get_reference_separator:
# WHEN: Calling get_reference_separators
result = get_reference_separators()
# THEN: The result should contain the 'verse', 'range', 'list' keys and get_reference_separator should have
# been called with the expected values.
assert all(key in result for key in ('verse', 'range', 'list')) is True
mocked_get_reference_separator.assert_has_calls(
[call('sep_v_display'), call('sep_r_display'), call('sep_l_display')])
def test_bible_search_enum():
"""
Test that the :class:`BibleSearch` class contains the expected enumerations
"""
# GIVEN: The BibleSearch class
# WHEN: Testing its attributes
# THEN: The BibleSearch class should have the following enumrations
assert hasattr(BibleSearch, 'Combined')
assert hasattr(BibleSearch, 'Reference')
assert hasattr(BibleSearch, 'Text')
def test_bible_media_item_subclass():
"""
Test that the :class:`BibleMediaItem` class is a subclass of the :class:`MediaManagerItem` class
"""
# GIVEN: The :class:`BibleMediaItem`
# WHEN: Checking if it is a subclass of MediaManagerItem
# THEN: BibleMediaItem should be a subclass of MediaManagerItem
assert issubclass(BibleMediaItem, MediaManagerItem)
def test_bible_media_item_signals():
"""
Test that the :class:`BibleMediaItem` class has the expected signals
"""
# GIVEN: The :class:`BibleMediaItem`
# THEN: The :class:`BibleMediaItem` should contain the following pyqtSignal's
assert hasattr(BibleMediaItem, 'bibles_go_live')
assert hasattr(BibleMediaItem, 'bibles_add_to_service')
assert isinstance(BibleMediaItem.bibles_go_live, QtCore.pyqtSignal)
assert isinstance(BibleMediaItem.bibles_add_to_service, QtCore.pyqtSignal)
def test_media_item_instance(media_item):
"""
When creating an instance of C test that it is also an instance of
:class:`MediaManagerItem`
"""
# GIVEN: An instance of :class:`BibleMediaItem`
# WEHN: Checking its class
# THEN: It should be a subclass of :class:`MediaManagerItem`
assert isinstance(media_item, MediaManagerItem)
def test_setup_item(media_item):
"""
Test the setup_item method
"""
# Could have tested the connection of the custom signals, however they're class vairables, and I could not find
# a way to properly test them.
# GIVEN: A mocked Registry.register_function method and an instance of BibleMediaItem
with patch.object(Registry(), 'register_function') as mocked_register_function:
# WHEN: Calling setup_itme
media_item.setup_item()
# THEN: Registry.register_function method should have been called with the reload_bibles method
mocked_register_function.assert_called_once_with('bibles_load_list', media_item.reload_bibles)
def test_required_icons(media_item):
"""
Test that all the required icons are set properly.
"""
# GIVEN: An instance of :class:`MediaManagerItem`
# WHEN: required_icons is called
media_item.required_icons()
# THEN: The correct icons should be set
assert media_item.has_import_icon is True, 'Check that the icon is as True.'
assert media_item.has_new_icon is False, 'Check that the icon is called as False.'
assert media_item.has_edit_icon is True, 'Check that the icon is called as True.'
assert media_item.has_delete_icon is True, 'Check that the icon is called as True.'
assert media_item.add_to_service_item is False, 'Check that the icon is called as False'
def test_on_focus_search_tab_visible(media_item):
"""
Test the correct widget gets focus when the BibleMediaItem receives focus
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out tabs and primary widgets
media_item.search_tab = MagicMock(**{'isVisible.return_value': True})
media_item.search_edit = MagicMock()
media_item.select_tab = MagicMock(**{'isVisible.return_value': False})
media_item.select_book_combo_box = MagicMock()
media_item.options_tab = MagicMock(**{'isVisible.return_value': False})
media_item.version_combo_box = MagicMock()
# WHEN: Calling on_focus
media_item.on_focus()
# THEN: search_edit should now have focus and its text selected
media_item.search_edit.assert_has_calls([call.setFocus(), call.selectAll()])
media_item.select_book_combo_box.assert_not_called()
media_item.version_combo_box.setFocus.assert_not_called()
def test_on_focus_select_tab_visible(media_item):
"""
Test the correct widget gets focus when the BibleMediaItem receives focus
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out tabs and primary widgets
media_item.search_tab = MagicMock(**{'isVisible.return_value': False})
media_item.search_edit = MagicMock()
media_item.select_tab = MagicMock(**{'isVisible.return_value': True})
media_item.select_book_combo_box = MagicMock()
media_item.options_tab = MagicMock(**{'isVisible.return_value': False})
media_item.version_combo_box = MagicMock()
# WHEN: Calling on_focus
media_item.on_focus()
# THEN: select_book_combo_box should have focus
media_item.search_edit.setFocus.assert_not_called()
media_item.search_edit.selectAll.assert_not_called()
media_item.select_book_combo_box.setFocus.assert_called_once_with()
media_item.version_combo_box.setFocus.assert_not_called()
def test_on_focus_options_tab_visible(media_item):
"""
Test the correct widget gets focus when the BibleMediaItem receives focus
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out tabs and primary widgets
media_item.search_tab = MagicMock(**{'isVisible.return_value': False})
media_item.search_edit = MagicMock()
media_item.select_tab = MagicMock(**{'isVisible.return_value': False})
media_item.select_book_combo_box = MagicMock()
media_item.options_tab = MagicMock(**{'isVisible.return_value': True})
media_item.version_combo_box = MagicMock()
# WHEN: Calling on_focus
media_item.on_focus()
# THEN: version_combo_box have received focus
media_item.search_edit.setFocus.assert_not_called()
media_item.search_edit.selectAll.assert_not_called()
media_item.select_book_combo_box.setFocus.assert_not_called()
media_item.version_combo_box.setFocus.assert_called_once_with()
def test_config_update_show_second_bible(media_item):
"""
Test the config update method
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out settings class with known values
media_item.settings.value = lambda key: {'bibles/second bibles': True}[key]
media_item.general_bible_layout = MagicMock()
media_item.second_combo_box = MagicMock()
# WHEN: Calling config_update()
media_item.config_update()
# THEN: second_combo_box() should be set visible
media_item.second_combo_box.setVisible.assert_called_once_with(True)
def test_config_update_hide_second_bible(media_item):
"""
Test the config update method
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out settings class with known values
media_item.settings.value = lambda key: {'bibles/second bibles': False}[key]
media_item.general_bible_layout = MagicMock()
media_item.second_combo_box = MagicMock()
# WHEN: Calling config_update()
media_item.config_update()
# THEN: second_combo_box() should hidden
media_item.second_combo_box.setVisible.assert_called_once_with(False)
def test_initalise(media_item):
"""
Test the initalise method
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out settings class with known values
media_item.settings.value = lambda key: {'bibles/reset to combined quick search': False}[key]
with patch.object(media_item, 'populate_bible_combo_boxes'), \
patch.object(media_item, 'config_update'):
media_item.search_edit = MagicMock()
# WHEN: Calling initialise()
media_item.initialise()
# THEN: The search_edit search types should have been set.
assert media_item.search_edit.set_search_types.called is True
assert media_item.search_edit.set_current_search_type.called is False
def test_initalise_reset_search_type(media_item):
"""
Test the initalise method
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out settings class with known values
media_item.settings.value = lambda key: {'bibles/reset to combined quick search': True}[key]
with patch.object(media_item, 'populate_bible_combo_boxes'), \
patch.object(media_item, 'config_update'):
media_item.search_edit = MagicMock()
# WHEN: Calling initialise()
media_item.initialise()
# THEN: The search_edit search types should have been set and that the current search type should be set to
# 'Combined'
assert media_item.search_edit.set_search_types.called is True
media_item.search_edit.set_current_search_type.assert_called_once_with(BibleSearch.Combined)
def test_populate_bible_combo_boxes(media_item):
"""
Test populate_bible_combo_boxes method
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out settings class with known values
bible_1 = MagicMock()
bible_2 = MagicMock()
bible_3 = MagicMock()
media_item.settings.value = lambda key: {'bibles/primary bible': bible_2, 'bibles/second bible': bible_3}[key]
media_item.version_combo_box = MagicMock()
media_item.second_combo_box = MagicMock()
media_item.plugin.manager.get_bibles.return_value = \
{'Bible 2': bible_2, 'Bible 1': bible_1, 'Bible 3': bible_3}
with patch('openlp.plugins.bibles.lib.mediaitem.get_locale_key', side_effect=lambda x: x), \
patch('openlp.plugins.bibles.lib.mediaitem.find_and_set_in_combo_box') as mocked_set_combo_box, \
patch.object(media_item, 'on_version_combo_box_index_changed'):
# WHEN: Calling populate_bible_combo_boxes
media_item.populate_bible_combo_boxes()
# THEN: The bible combo boxes should be filled with the bible names and data, in a sorted order.
# Also check the correct combo box has been selected according to the settings
media_item.version_combo_box.addItem.assert_has_calls(
[call('Bible 1', bible_1), call('Bible 2', bible_2), call('Bible 3', bible_3)])
media_item.second_combo_box.addItem.assert_has_calls(
[call('', None), call('Bible 1', bible_1), call('Bible 2', bible_2), call('Bible 3', bible_3)])
mocked_set_combo_box.assert_has_calls(
[call(media_item.version_combo_box, bible_2), call(media_item.second_combo_box, bible_3)])
def test_reload_bibles(media_item):
"""
Test reload_bibles
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out settings class with known values
with patch.object(media_item, 'populate_bible_combo_boxes') as mocked_populate_bible_combo_boxes:
# WHEN: Calling reload_bibles()
media_item.reload_bibles()
# THEN: The manager reload_bibles method should have been called and the bible combo boxes updated
media_item.plugin.manager.reload_bibles.assert_called_once_with()
mocked_populate_bible_combo_boxes.assert_called_once_with()
def test_get_common_books_no_second_book(media_item):
"""
Test get_common_books when called with out a second bible
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked first bible
# WHEN: Calling get_common_books with only one bible
result = media_item.get_common_books(mocked_bible_1)
# THEN: The book of the bible should be returned
assert result == book_list_1
def test_get_common_books_second_book(media_item):
"""
Test get_common_books when called with a second bible
"""
# GIVEN: An instance of :class:`MediaManagerItem` and two mocked bibles with differing books
# WHEN: Calling get_common_books with two bibles
result = media_item.get_common_books(mocked_bible_1, mocked_bible_2)
# THEN: Only the books contained in both bibles should be returned
assert result == [mocked_book_2, mocked_book_3]
def test_initialise_advanced_bible_no_bible(media_item):
"""
Test initialise_advanced_bible when there is no main bible
"""
# GIVEN: An instance of :class:`MediaManagerItem`
media_item.select_book_combo_box = MagicMock()
with patch.object(media_item, 'get_common_books') as mocked_get_common_books:
# WHEN: Calling initialise_advanced_bible() when there is no main bible
media_item.bible = None
result = media_item.initialise_advanced_bible()
# THEN: initialise_advanced_bible should return with put calling get_common_books
assert result is None
mocked_get_common_books.assert_not_called()
def test_initialise_advanced_bible_add_books_with_last_id_found(media_item):
"""
Test initialise_advanced_bible when the last_id argument is supplied and it is found in the list
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked_book_combo_box which simulates data being found
# in the list
media_item.select_book_combo_box = MagicMock(**{'findData.return_value': 2})
with patch.object(media_item, 'get_common_books', return_value=book_list_1), \
patch.object(media_item, 'on_advanced_book_combo_box'):
# WHEN: Calling initialise_advanced_bible() with the last_id argument set
media_item.bible = MagicMock()
media_item.initialise_advanced_bible(10)
# THEN: The books should be added to the combo box, and the chosen book should be reselected
media_item.select_book_combo_box.addItem.assert_has_calls(
[call('Book 1', 1), call('Book 2', 2), call('Book 3', 3)])
media_item.select_book_combo_box.setCurrentIndex.assert_called_once_with(2)
def test_initialise_advanced_bible_add_books_with_last_id_not_found(media_item):
"""
Test initialise_advanced_bible when the last_id argument is supplied and it is not found in the list
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked_book_combo_box which simulates data not being
# found in the list
media_item.select_book_combo_box = MagicMock(**{'findData.return_value': -1})
with patch.object(media_item, 'get_common_books', return_value=book_list_1), \
patch.object(media_item, 'on_advanced_book_combo_box'):
# WHEN: Calling initialise_advanced_bible() with the last_id argument set
media_item.bible = MagicMock()
media_item.initialise_advanced_bible(10)
# THEN: The books should be added to the combo box, and the first book should be selected
media_item.select_book_combo_box.addItem.assert_has_calls(
[call('Book 1', 1), call('Book 2', 2), call('Book 3', 3)])
media_item.select_book_combo_box.setCurrentIndex.assert_called_once_with(0)
def test_update_auto_completer_search_no_bible(media_item):
"""
Test update_auto_completer when there is no main bible selected and the search_edit type is
'BibleSearch.Reference'
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked search_edit
mocked_search_edit = MagicMock(**{'current_search_type.return_value': BibleSearch.Reference})
media_item.search_edit = mocked_search_edit
media_item.bible = None
with patch.object(media_item, 'get_common_books') as mocked_get_common_books, \
patch('openlp.plugins.bibles.lib.mediaitem.set_case_insensitive_completer') \
as mocked_set_case_insensitive_completer:
# WHEN: Calling update_auto_completer
media_item.update_auto_completer()
# THEN: get_common_books should not have been called. set_case_insensitive_completer should have been called
# with an empty list
mocked_get_common_books.assert_not_called()
mocked_set_case_insensitive_completer.assert_called_once_with([], mocked_search_edit)
def test_update_auto_completer_search_reference_type(media_item):
"""
Test update_auto_completer when a main bible is selected and the search_edit type is 'BibleSearch.Reference'
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked search_edit
mocked_search_edit = MagicMock(**{'current_search_type.return_value': BibleSearch.Reference})
media_item.search_edit = mocked_search_edit
media_item.bible = MagicMock()
with patch.object(media_item, 'get_common_books', return_value=book_list_1), \
patch('openlp.plugins.bibles.lib.mediaitem.get_locale_key', side_effect=lambda x: x), \
patch('openlp.plugins.bibles.lib.mediaitem.set_case_insensitive_completer') \
as mocked_set_case_insensitive_completer:
# WHEN: Calling update_auto_completer
media_item.update_auto_completer()
# THEN: set_case_insensitive_completer should have been called with the names of the books + space in order
mocked_set_case_insensitive_completer.assert_called_once_with(
['Book 1 ', 'Book 2 ', 'Book 3 '], mocked_search_edit)
def test_update_auto_completer_search_combined_type(media_item):
"""
Test update_auto_completer when a main bible is selected and the search_edit type is 'BibleSearch.Combined'
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked search_edit
mocked_search_edit = MagicMock(**{'current_search_type.return_value': BibleSearch.Combined})
media_item.search_edit = mocked_search_edit
media_item.bible = MagicMock()
with patch.object(media_item, 'get_common_books', return_value=book_list_1), \
patch('openlp.plugins.bibles.lib.mediaitem.get_locale_key', side_effect=lambda x: x), \
patch('openlp.plugins.bibles.lib.mediaitem.set_case_insensitive_completer') \
as mocked_set_case_insensitive_completer:
# WHEN: Calling update_auto_completer
media_item.update_auto_completer()
# THEN: set_case_insensitive_completer should have been called with the names of the books + space in order
mocked_set_case_insensitive_completer.assert_called_once_with(
['Book 1 ', 'Book 2 ', 'Book 3 '], mocked_search_edit)
def test_on_import_click_no_import_wizard_attr(media_item):
"""
Test on_import_click when media_item does not have the `import_wizard` attribute. And the wizard was canceled.
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked BibleImportForm
mocked_bible_import_form_instance = MagicMock(**{'exec.return_value': False})
with patch('openlp.plugins.bibles.lib.mediaitem.BibleImportForm',
return_value=mocked_bible_import_form_instance) as mocked_bible_import_form, \
patch.object(media_item, 'reload_bibles') as mocked_reload_bibles:
# WHEN: Calling on_import_click
media_item.on_import_click()
# THEN: BibleImport wizard should have been instianed and reload_bibles should not have been called
assert mocked_bible_import_form.called is True
assert mocked_reload_bibles.called is False
def test_on_import_click_wizard_not_canceled(media_item):
"""
Test on_import_click when the media item has the import_wizard attr set and wizard completes sucessfully.
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked import_wizard
mocked_import_wizard = MagicMock(**{'exec.return_value': True})
media_item.import_wizard = mocked_import_wizard
with patch.object(media_item, 'reload_bibles') as mocked_reload_bibles:
# WHEN: Calling on_import_click
media_item.on_import_click()
# THEN: BibleImport wizard should have been instianted and reload_bibles should not have been called
assert mocked_import_wizard.called is False
assert mocked_reload_bibles.called is True
def test_on_edit_click_no_bible(media_item):
"""
Test on_edit_click when there is no main bible selected
"""
# GIVEN: An instance of :class:`MediaManagerItem`
with patch('openlp.plugins.bibles.lib.mediaitem.EditBibleForm') as mocked_edit_bible_form:
# WHEN: A main bible is not selected and on_edit_click is called
media_item.bible = None
media_item.on_edit_click()
# THEN: EditBibleForm should not have been instianted
assert mocked_edit_bible_form.called is False
def test_on_edit_click_user_cancel_edit_form(media_item):
"""
Test on_edit_click when the user cancels the EditBibleForm
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked EditBibleForm which returns False when exec is
# called
media_item.bible = MagicMock()
mocked_edit_bible_form_instance = MagicMock(**{'exec.return_value': False})
with patch('openlp.plugins.bibles.lib.mediaitem.EditBibleForm', return_value=mocked_edit_bible_form_instance) \
as mocked_edit_bible_form, \
patch.object(media_item, 'reload_bibles') as mocked_reload_bibles:
# WHEN: on_edit_click is called, and the user cancels the EditBibleForm
media_item.on_edit_click()
# THEN: EditBibleForm should have been been instianted but reload_bibles should not have been called
assert mocked_edit_bible_form.called is True
assert mocked_reload_bibles.called is False
def test_on_edit_click_user_accepts_edit_form(media_item):
"""
Test on_edit_click when the user accepts the EditBibleForm
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked EditBibleForm which returns True when exec is
# called
media_item.bible = MagicMock()
mocked_edit_bible_form_instance = MagicMock(**{'exec.return_value': True})
with patch('openlp.plugins.bibles.lib.mediaitem.EditBibleForm',
return_value=mocked_edit_bible_form_instance) \
as mocked_edit_bible_form, \
patch.object(media_item, 'reload_bibles') as mocked_reload_bibles:
# WHEN: on_edit_click is called, and the user accpets the EditBibleForm
media_item.on_edit_click()
# THEN: EditBibleForm should have been been instianted and reload_bibles should have been called
assert mocked_edit_bible_form.called is True
assert mocked_reload_bibles.called is True
def test_on_delete_click_no_bible(media_item):
"""
Test on_delete_click when there is no main bible selected
"""
# GIVEN: An instance of :class:`MediaManagerItem`
with patch('openlp.plugins.bibles.lib.mediaitem.QtWidgets.QMessageBox') as mocked_qmessage_box:
# WHEN: A main bible is not selected and on_delete_click is called
media_item.bible = None
media_item.on_delete_click()
# THEN: QMessageBox.question should not have been called
assert mocked_qmessage_box.question.called is False
def test_on_delete_click_response_no(media_item):
"""
Test on_delete_click when the user selects no from the message box
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a QMessageBox which reutrns QtWidgets.QMessageBox.No
media_item.bible = MagicMock()
with patch('openlp.plugins.bibles.lib.mediaitem.QtWidgets.QMessageBox.question',
return_value=QtWidgets.QMessageBox.No) as mocked_qmessage_box:
# WHEN: on_delete_click is called
media_item.on_delete_click()
# THEN: QMessageBox.question should have been called, but the delete_bible should not have been called
assert mocked_qmessage_box.called is True
assert media_item.plugin.manager.delete_bible.called is False
def test_on_delete_click_response_yes(media_item):
"""
Test on_delete_click when the user selects yes from the message box
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a QMessageBox which reutrns QtWidgets.QMessageBox.Yes
media_item.bible = MagicMock()
with patch('openlp.plugins.bibles.lib.mediaitem.QtWidgets.QMessageBox.question',
return_value=QtWidgets.QMessageBox.Yes) as mocked_qmessage_box, \
patch.object(media_item, 'reload_bibles'):
# WHEN: on_delete_click is called
media_item.on_delete_click()
# THEN: QMessageBox.question should and delete_bible should not have been called
assert mocked_qmessage_box.called is True
assert media_item.plugin.manager.delete_bible.called is True
def test_on_search_tab_bar_current_changed_search_tab_selected(media_item):
"""
Test on_search_tab_bar_current_changed when the search_tab is selected
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out search_tab and select_tab
media_item.search_tab = MagicMock()
media_item.select_tab = MagicMock()
media_item.options_tab = MagicMock()
media_item.search_button = MagicMock()
with patch.object(media_item, 'on_focus'):
# WHEN: The search_tab has been selected
media_item.on_search_tab_bar_current_changed(SearchTabs.Search)
# THEN: The search_button should be enabled, search_tab should be setVisible and select_tab should be hidden
media_item.search_button.setEnabled.assert_called_once_with(True)
media_item.search_tab.setVisible.assert_called_once_with(True)
media_item.select_tab.setVisible.assert_called_once_with(False)
def test_on_search_tab_bar_current_changed_select_tab_selected(media_item):
"""
Test on_search_tab_bar_current_changed when the select_tab is selected
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out search_tab and select_tab
media_item.search_tab = MagicMock()
media_item.select_tab = MagicMock()
media_item.options_tab = MagicMock()
media_item.search_button = MagicMock()
with patch.object(media_item, 'on_focus'):
# WHEN: The select_tab has been selected
media_item.on_search_tab_bar_current_changed(SearchTabs.Select)
# THEN: The search_button should be enabled, select_tab should be setVisible and search_tab should be hidden
media_item.search_button.setEnabled.assert_called_once_with(True)
media_item.search_tab.setVisible.assert_called_once_with(False)
media_item.select_tab.setVisible.assert_called_once_with(True)
def test_on_book_order_button_toggled_checked(media_item):
"""
Test that 'on_book_order_button_toggled' changes the order of the book list
"""
media_item.select_book_combo_box = MagicMock()
# WHEN: When the book_order_button is checked
media_item.on_book_order_button_toggled(True)
# THEN: The select_book_combo_box model should have been sorted
media_item.select_book_combo_box.model().sort.assert_called_once_with(0)
def test_on_book_order_button_toggled_un_checked(media_item):
"""
Test that 'on_book_order_button_toggled' changes the order of the book list
"""
media_item.select_book_combo_box = MagicMock()
# WHEN: When the book_order_button is un-checked
media_item.on_book_order_button_toggled(False)
# THEN: The select_book_combo_box model sort should have been reset
media_item.select_book_combo_box.model().sort.assert_called_once_with(-1)
def test_on_clear_button_clicked(media_item):
"""
Test on_clear_button_clicked when the search tab is selected
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked out search_tab and select_tab and a mocked out
# list_view and search_edit
media_item.list_view = MagicMock(**{'selectedItems.return_value': ['Some', 'Results']})
media_item.results_view_tab = MagicMock(**{'currentIndex.return_value': ResultsTab.Search})
with patch.object(media_item, 'on_results_view_tab_total_update'):
# WHEN: Calling on_clear_button_clicked
media_item.on_clear_button_clicked()
# THEN: The list_view and the search_edit should be cleared
assert media_item.current_results == []
assert media_item.list_view.takeItem.call_count == 2
media_item.list_view.row.assert_has_calls([call('Some'), call('Results')])
def test_on_save_results_button_clicked(media_item):
"""
Test that "on_save_results_button_clicked" saves the results.
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked list_view
result_1 = MagicMock(**{'data.return_value': 'R1'})
result_2 = MagicMock(**{'data.return_value': 'R2'})
result_3 = MagicMock(**{'data.return_value': 'R3'})
media_item.list_view = MagicMock(**{'selectedItems.return_value': [result_1, result_2, result_3]})
with patch.object(media_item, 'on_results_view_tab_total_update') as \
mocked_on_results_view_tab_total_update:
# WHEN: When the save_results_button is clicked
media_item.on_save_results_button_clicked()
# THEN: The selected results in the list_view should be added to the 'saved_results' list. And the saved_tab
# total should be updated.
assert media_item.saved_results == ['R1', 'R2', 'R3']
mocked_on_results_view_tab_total_update.assert_called_once_with(ResultsTab.Saved)
def test_on_style_combo_box_changed(media_item):
"""
Test on_style_combo_box_index_changed
"""
# GIVEN: An instance of :class:`MediaManagerItem` a mocked media_item.settings
media_item.settings_tab = MagicMock()
# WHEN: Calling on_style_combo_box_index_changed
media_item.on_style_combo_box_index_changed(2)
# THEN: The layout_style setting should have been set
assert media_item.settings_tab.layout_style == 2
def test_on_version_combo_box_index_changed_no_bible(media_item):
"""
Test on_version_combo_box_index_changed when there is no main bible.
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked media_item.settings and select_book_combo_box
media_item.version_combo_box = MagicMock(**{'currentData.return_value': None})
media_item.select_book_combo_box = MagicMock()
with patch.object(media_item, 'initialise_advanced_bible'):
# WHEN: Calling on_version_combo_box_index_changed
media_item.on_version_combo_box_index_changed()
# THEN: The version should be saved to settings and the 'select tab' should be initialised
assert media_item.initialise_advanced_bible.called is True
def test_on_version_combo_box_index_changed_bible_selected(media_item):
"""
Test on_version_combo_box_index_changed when a bible has been selected.
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked media_item.settings and select_book_combo_box
mocked_bible_db = MagicMock()
mocked_bible_db.name = 'ABC'
media_item.version_combo_box = MagicMock(**{'currentData.return_value': mocked_bible_db})
media_item.select_book_combo_box = MagicMock()
with patch.object(media_item, 'initialise_advanced_bible'):
# WHEN: Calling on_version_combo_box_index_changed
media_item.on_version_combo_box_index_changed()
# THEN: The version should be saved to settings and the 'select tab' should be initialised
assert media_item.initialise_advanced_bible.called is True
def test_on_second_combo_box_index_changed_mode_not_changed(media_item):
"""
Test on_second_combo_box_index_changed when the user does not change from dual mode
results and the user chooses no to the message box
"""
# GIVEN: An instance of :class:`MediaManagerItem`
media_item.list_view = MagicMock(**{'count.return_value': 5})
media_item.style_combo_box = MagicMock()
media_item.select_book_combo_box = MagicMock()
with patch.object(media_item, 'initialise_advanced_bible'), \
patch('openlp.plugins.bibles.lib.mediaitem.critical_error_message_box') \
as mocked_critical_error_message_box:
# WHEN: The previously selected bible is one bible and the new selection is another bible
media_item.second_bible = mocked_bible_1
media_item.second_combo_box = MagicMock(**{'currentData.return_value': mocked_bible_2})
media_item.on_second_combo_box_index_changed(5)
# THEN: The new bible should now be the current bible
assert mocked_critical_error_message_box.called is False
media_item.style_combo_box.setEnabled.assert_called_once_with(False)
assert media_item.second_bible == mocked_bible_2
media_item.settings.setValue.assert_called_once_with('bibles/second bible', 'Bible 2')
def test_on_second_combo_box_index_changed_single_to_dual_user_abort(media_item):
"""
Test on_second_combo_box_index_changed when the user changes from single to dual bible mode, there are search
results and the user chooses no to the message box
"""
# GIVEN: An instance of :class:`MediaManagerItem`
media_item.list_view = MagicMock(**{'count.return_value': 5})
media_item.style_combo_box = MagicMock()
media_item.select_book_combo_box = MagicMock()
with patch.object(media_item, 'initialise_advanced_bible'), \
patch('openlp.plugins.bibles.lib.mediaitem.critical_error_message_box',
return_value=QtWidgets.QMessageBox.No) as mocked_critical_error_message_box:
# WHEN: The previously selected bible is None and the new selection is a bible and the user selects yes
# to the dialog box
media_item.second_bible = None
media_item.second_combo_box = MagicMock(**{'currentData.return_value': mocked_bible_1})
media_item.saved_results = ['saved_results']
media_item.on_second_combo_box_index_changed(5)
# THEN: The list_view should be cleared and the currently selected bible should not be changed
assert mocked_critical_error_message_box.called is True
assert media_item.second_combo_box.setCurrentIndex.called is True
assert media_item.style_combo_box.setEnabled.called is False
assert media_item.second_bible is None
media_item.settings.setValue.assert_not_called()
def test_on_second_combo_box_index_changed_single_to_dual(media_item):
"""
Test on_second_combo_box_index_changed when the user changes from single to dual bible mode, there are search
results and the user chooses yes to the message box
"""
# GIVEN: An instance of :class:`MediaManagerItem`
media_item.list_view = MagicMock(**{'count.return_value': 5})
media_item.style_combo_box = MagicMock()
media_item.select_book_combo_box = MagicMock()
media_item.search_results = ['list', 'of', 'results']
with patch.object(media_item, 'initialise_advanced_bible') as mocked_initialise_advanced_bible, \
patch.object(media_item, 'display_results'), \
patch('openlp.plugins.bibles.lib.mediaitem.critical_error_message_box',
return_value=QtWidgets.QMessageBox.Yes) as mocked_critical_error_message_box:
# WHEN: The previously selected bible is None and the new selection is a bible and the user selects yes
# to the dialog box
media_item.second_bible = None
media_item.second_combo_box = MagicMock(**{'currentData.return_value': mocked_bible_1})
media_item.saved_results = ['saved_results']
media_item.on_second_combo_box_index_changed(5)
# THEN: The selected bible should be set as the current bible
assert mocked_critical_error_message_box.called is True
media_item.style_combo_box.setEnabled.assert_called_once_with(False)
assert mocked_initialise_advanced_bible.called is True
assert media_item.second_bible == mocked_bible_1
media_item.settings.setValue.assert_called_once_with('bibles/second bible', 'Bible 1')
def test_on_second_combo_box_index_changed_dual_to_single(media_item):
"""
Test on_second_combo_box_index_changed when the user changes from dual to single bible mode, there are search
results and the user chooses yes to the message box
"""
# GIVEN: An instance of :class:`MediaManagerItem`
media_item.list_view = MagicMock(**{'count.return_value': 5})
media_item.style_combo_box = MagicMock()
media_item.select_book_combo_box = MagicMock()
media_item.search_results = ['list', 'of', 'results']
with patch.object(media_item, 'initialise_advanced_bible') as mocked_initialise_advanced_bible, \
patch.object(media_item, 'display_results'), \
patch('openlp.plugins.bibles.lib.mediaitem.critical_error_message_box',
return_value=QtWidgets.QMessageBox.Yes) as mocked_critical_error_message_box:
# WHEN: The previously is a bible new selection is None and the user selects yes
# to the dialog box
media_item.second_bible = mocked_bible_1
media_item.second_combo_box = MagicMock(**{'currentData.return_value': None})
media_item.saved_results = ['saved_results']
media_item.on_second_combo_box_index_changed(0)
# THEN: The selected bible should be set as the current bible
assert mocked_critical_error_message_box.called is True
media_item.style_combo_box.setEnabled.assert_called_once_with(True)
assert mocked_initialise_advanced_bible.called is False
assert media_item.second_bible is None
media_item.settings.setValue.assert_called_once_with('bibles/second bible', None)
def test_on_advanced_book_combo_box(media_item):
"""
Test on_advanced_book_combo_box when the book returns 0 for the verse count.
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked get_verse_count_by_book_ref_id which returns 0
media_item.select_book_combo_box = MagicMock(**{'currentData.return_value': 2})
media_item.bible = mocked_bible_1
media_item.plugin.manager.get_verse_count_by_book_ref_id.return_value = 0
media_item.search_button = MagicMock()
with patch('openlp.plugins.bibles.lib.mediaitem.critical_error_message_box') \
as mocked_critical_error_message_box:
# WHEN: Calling on_advanced_book_combo_box
media_item.on_advanced_book_combo_box()
# THEN: The user should be informed that the bible cannot be used and the search button should be disabled
media_item.plugin.manager.get_book_by_id.assert_called_once_with('Bible 1', 2)
media_item.search_button.setEnabled.assert_called_once_with(False)
assert mocked_critical_error_message_box.called is True
def test_on_advanced_book_combo_box_no_selection(media_item):
"""
Test on_advanced_book_combo_box when there is nothing selected
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked manager
media_item.select_book_combo_box = MagicMock(**{'currentData.return_value': None})
# WHEN: Calling on_advanced_book_combo_box
media_item.on_advanced_book_combo_box()
# THEN: The method should exit early, and no query made
assert media_item.plugin.manager.get_book_by_id.call_count == 0
def test_on_advanced_book_combo_box_set_up_comboboxes(media_item):
"""
Test on_advanced_book_combo_box when the book returns 6 for the verse count.
"""
# GIVEN: An instance of :class:`MediaManagerItem` and a mocked get_verse_count_by_book_ref_id which returns 6
media_item.from_chapter = 0
media_item.to_chapter = 0
media_item.from_verse = 0
media_item.to_verse = 0
media_item.select_book_combo_box = MagicMock(**{'currentData.return_value': 2})
media_item.bible = mocked_bible_1
media_item.plugin.manager.get_verse_count_by_book_ref_id.return_value = 6
media_item.select_tab = MagicMock(**{'isVisible.return_value': True})
media_item.search_button = MagicMock()
with patch.object(media_item, 'adjust_combo_box') as mocked_adjust_combo_box:
# WHEN: Calling on_advanced_book_combo_box
media_item.on_advanced_book_combo_box()
# THEN: The verse selection combobox's should be set up
media_item.plugin.manager.get_book_by_id.assert_called_once_with('Bible 1', 2)
media_item.search_button.setEnabled.assert_called_once_with(True)
assert mocked_adjust_combo_box.call_count == 4
def test_on_from_chapter_activated_invalid_to_chapter(media_item):
"""
Test on_from_chapter_activated when the to_chapter is less than the from_chapter
"""
# GIVEN: An instance of :class:`MediaManagerItem`, some mocked comboboxes with test data
media_item.chapter_count = 25
media_item.bible = mocked_bible_1
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock(**{'currentData.return_value': 10})
media_item.to_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.from_verse = MagicMock()
media_item.to_verse = MagicMock()
media_item.plugin.manager.get_verse_count_by_book_ref_id.return_value = 20
with patch.object(media_item, 'adjust_combo_box') as mocked_adjust_combo_box:
# WHEN: Calling on_from_chapter_activated
media_item.on_from_chapter_activated()
# THEN: The to_verse and to_chapter comboboxes should be updated appropriately
assert mocked_adjust_combo_box.call_args_list == [
call(1, 20, media_item.from_verse), call(1, 20, media_item.to_verse, False),
call(10, 25, media_item.to_chapter, False)]
def test_on_from_chapter_activated_same_chapter(media_item):
"""
Test on_from_chapter_activated when the to_chapter is the same as from_chapter
"""
# GIVEN: An instance of :class:`MediaManagerItem`, some mocked comboboxes with test data
media_item.chapter_count = 25
media_item.bible = mocked_bible_1
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.to_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.from_verse = MagicMock()
media_item.to_verse = MagicMock()
media_item.plugin.manager.get_verse_count_by_book_ref_id.return_value = 20
with patch.object(media_item, 'adjust_combo_box') as mocked_adjust_combo_box:
# WHEN: Calling on_from_chapter_activated
media_item.on_from_chapter_activated()
# THEN: The to_verse and to_chapter comboboxes should be updated appropriately
assert mocked_adjust_combo_box.call_args_list == [
call(1, 20, media_item.from_verse), call(1, 20, media_item.to_verse, True),
call(5, 25, media_item.to_chapter, False)]
def test_on_from_chapter_activated_lower_chapter(media_item):
"""
Test on_from_chapter_activated when the to_chapter is greater than the from_chapter
"""
# GIVEN: An instance of :class:`MediaManagerItem`, some mocked comboboxes with test data
media_item.chapter_count = 25
media_item.bible = mocked_bible_1
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.to_chapter = MagicMock(**{'currentData.return_value': 7})
media_item.from_verse = MagicMock()
media_item.to_verse = MagicMock()
media_item.plugin.manager.get_verse_count_by_book_ref_id.return_value = 20
with patch.object(media_item, 'adjust_combo_box') as mocked_adjust_combo_box:
# WHEN: Calling on_from_chapter_activated
media_item.on_from_chapter_activated()
# THEN: The to_verse and to_chapter comboboxes should be updated appropriately
assert mocked_adjust_combo_box.call_args_list == [
call(1, 20, media_item.from_verse), call(5, 25, media_item.to_chapter, True)]
def test_on_from_verse(media_item):
"""
Test on_from_verse when the to_chapter is not equal to the from_chapter
"""
# GIVEN: An instance of :class:`MediaManagerItem`, some mocked comboboxes with test data
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock(**{'currentData.return_value': 2})
media_item.to_chapter = MagicMock(**{'currentData.return_value': 5})
# WHEN: Calling on_from_verse
media_item.on_from_verse()
# THEN: select_book_combo_box.currentData should nto be called
assert media_item.select_book_combo_box.currentData.called is False
def test_on_from_verse_equal(media_item):
"""
Test on_from_verse when the to_chapter is equal to the from_chapter
"""
# GIVEN: An instance of :class:`MediaManagerItem`, some mocked comboboxes with test data
media_item.bible = mocked_bible_1
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.to_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.from_verse = MagicMock(**{'currentData.return_value': 7})
media_item.to_verse = MagicMock()
media_item.plugin.manager.get_verse_count_by_book_ref_id.return_value = 20
with patch.object(media_item, 'adjust_combo_box') as mocked_adjust_combo_box:
# WHEN: Calling on_from_verse
media_item.on_from_verse()
# THEN: The to_verse should have been updated
mocked_adjust_combo_box.assert_called_once_with(7, 20, media_item.to_verse, True)
def test_on_to_chapter_same_chapter_from_greater_than(media_item):
"""
Test on_to_chapter when the to_chapter is equal to the from_chapter and the from_verse is greater than the
to_verse
"""
# GIVEN: An instance of :class:`MediaManagerItem`, some mocked comboboxes with test data
media_item.bible = mocked_bible_1
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.to_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.from_verse = MagicMock(**{'currentData.return_value': 10})
media_item.to_verse = MagicMock(**{'currentData.return_value': 7})
media_item.plugin.manager.get_verse_count_by_book_ref_id.return_value = 20
with patch.object(media_item, 'adjust_combo_box') as mocked_adjust_combo_box:
# WHEN: Calling on_tp_chapter
media_item.on_to_chapter()
# THEN: The to_verse should have been updated
mocked_adjust_combo_box.assert_called_once_with(10, 20, media_item.to_verse)
def test_on_from_verse_chapters_not_equal(media_item):
"""
Test on_from_verse when the to_chapter is not equal to the from_chapter
"""
# GIVEN: An instance of :class:`MediaManagerItem`, some mocked comboboxes with test data
media_item.bible = mocked_bible_1
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock(**{'currentData.return_value': 7})
media_item.to_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.from_verse = MagicMock(**{'currentData.return_value': 10})
media_item.to_verse = MagicMock(**{'currentData.return_value': 7})
media_item.plugin.manager.get_verse_count_by_book_ref_id.return_value = 20
with patch.object(media_item, 'adjust_combo_box') as mocked_adjust_combo_box:
# WHEN: Calling on_from_chapter_activated
media_item.on_to_chapter()
# THEN: The to_verse should have been updated
mocked_adjust_combo_box.assert_called_once_with(1, 20, media_item.to_verse)
def test_on_from_verse_from_verse_less_than(media_item):
"""
Test on_from_verse when the to_chapter is equal to the from_chapter and from_verse is less than to_verse
"""
# GIVEN: An instance of :class:`MediaManagerItem`, some mocked comboboxes with test data
media_item.bible = mocked_bible_1
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.to_chapter = MagicMock(**{'currentData.return_value': 5})
media_item.from_verse = MagicMock(**{'currentData.return_value': 6})
media_item.to_verse = MagicMock(**{'currentData.return_value': 7})
media_item.plugin.manager.get_verse_count_by_book_ref_id.return_value = 20
with patch.object(media_item, 'adjust_combo_box') as mocked_adjust_combo_box:
# WHEN: Calling on_from_chapter_activated
media_item.on_to_chapter()
# THEN: The to_verse should have been updated
mocked_adjust_combo_box.assert_called_once_with(1, 20, media_item.to_verse)
def test_adjust_combo_box_no_restore(media_item):
"""
Test adjust_combo_box when being used with out the restore function
"""
# GIVEN: An instance of :class:`MediaManagerItem`
mocked_combo_box = MagicMock()
# WHEN: Calling adjust_combo_box with out setting the kwarg `restore`
media_item.adjust_combo_box(10, 13, mocked_combo_box)
# THEN: The combo_box should be cleared, and new items added
mocked_combo_box.clear.assert_called_once_with()
assert mocked_combo_box.addItem.call_args_list == \
[call('10', 10), call('11', 11), call('12', 12), call('13', 13)]
def test_adjust_combo_box_restore_found(media_item):
"""
Test adjust_combo_box when being used with out the restore function
"""
# GIVEN: An instance of :class:`MediaManagerItem`, with the 2nd item '12' selected
mocked_combo_box = MagicMock(**{'currentData.return_value': 12, 'findData.return_value': 2})
# WHEN: Calling adjust_combo_box with the kwarg `restore` set to True
media_item.adjust_combo_box(10, 13, mocked_combo_box, True)
# THEN: The combo_box should be cleared, and new items added. Finally the previously selected item should be
# reselected
mocked_combo_box.clear.assert_called_once_with()
assert mocked_combo_box.addItem.call_args_list == \
[call('10', 10), call('11', 11), call('12', 12), call('13', 13)]
mocked_combo_box.setCurrentIndex.assert_called_once_with(2)
def test_adjust_combo_box_restore_not_found(media_item):
"""
Test adjust_combo_box when being used with out the restore function when the selected item is not available
after the combobox has been updated
"""
# GIVEN: An instance of :class:`MediaManagerItem`, with the 2nd item '12' selected
mocked_combo_box = MagicMock(**{'currentData.return_value': 9, 'findData.return_value': -1})
# WHEN: Calling adjust_combo_box with the kwarg `restore` set to True
media_item.adjust_combo_box(10, 13, mocked_combo_box, True)
# THEN: The combo_box should be cleared, and new items added. Finally the first item should be selected
mocked_combo_box.clear.assert_called_once_with()
assert mocked_combo_box.addItem.call_args_list == \
[call('10', 10), call('11', 11), call('12', 12), call('13', 13)]
mocked_combo_box.setCurrentIndex.assert_called_once_with(0)
def test_on_search_button_no_bible(media_item):
"""
Test on_search_button_clicked when there is no bible selected
"""
# GIVEN: An instance of :class:`MediaManagerItem`
# WHEN calling on_search_button_clicked and there is no selected bible
media_item.bible = None
media_item.on_search_button_clicked()
# THEN: The user should be informed that there are no bibles selected
assert Registry().get('main_window').information_message.call_count == 1
def test_on_search_button_search_tab(media_item):
"""
Test on_search_button_clicked when the `Search` tab is selected
"""
# GIVEN: An instance of :class:`MediaManagerItem`, and a mocked text_search method
media_item.bible = mocked_bible_1
media_item.search_button = MagicMock()
media_item.search_tab = MagicMock(**{'isVisible.return_value': True})
with patch.object(media_item, 'text_search') as mocked_text_search:
# WHEN: Calling on_search_button_clicked and the 'Search' tab is selected
media_item.on_search_button_clicked()
# THEN: The text_search method should have been called
mocked_text_search.assert_called_once_with()
def test_on_search_button_select_tab(media_item):
"""
Test on_search_button_clicked when the `Select` tab is selected
"""
# GIVEN: An instance of :class:`MediaManagerItem`, and a mocked select_search method
media_item.bible = mocked_bible_1
media_item.search_button = MagicMock()
media_item.search_tab = MagicMock(**{'isVisible.return_value': False})
media_item.select_tab = MagicMock(**{'isVisible.return_value': True})
with patch.object(media_item, 'select_search') as mocked_select_search:
# WHEN: Calling on_search_button_clicked and the 'Select' tab is selected
media_item.on_search_button_clicked()
# THEN: The text_search method should have been called
mocked_select_search.assert_called_once_with()
def test_select_search_single_bible(media_item):
"""
Test select_search when only one bible is selected
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked plugin.manager.get_verses
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock()
media_item.from_verse = MagicMock()
media_item.to_chapter = MagicMock()
media_item.to_verse = MagicMock()
with patch.object(media_item, 'display_results') as mocked_display_results:
# WHEN: Calling select_search and there is only one bible selected
media_item.bible = mocked_bible_1
media_item.second_bible = None
media_item.select_search()
# THEN: reference_search should only be called once
assert media_item.plugin.manager.get_verses.call_count == 1
mocked_display_results.assert_called_once_with()
def test_select_search_dual_bibles(media_item):
"""
Test select_search when two bibles are selected
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked_reference_search
media_item.select_book_combo_box = MagicMock()
media_item.from_chapter = MagicMock()
media_item.from_verse = MagicMock()
media_item.to_chapter = MagicMock()
media_item.to_verse = MagicMock()
with patch.object(media_item, 'display_results') as mocked_display_results:
# WHEN: Calling select_search and there are two bibles selected
media_item.bible = mocked_bible_1
media_item.second_bible = mocked_bible_2
media_item.select_search()
# THEN: reference_search should be called twice
assert media_item.plugin.manager.get_verses.call_count == 2
mocked_display_results.assert_called_once_with()
def test_text_reference_search_single_bible(media_item):
"""
Test text_reference_search when only one bible is selected
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked plugin.manager.get_verses
with patch.object(media_item, 'display_results') as mocked_display_results:
# WHEN: Calling text_reference_search with only one bible selected
media_item.bible = mocked_bible_1
media_item.second_bible = None
media_item.text_reference_search('Search Text')
# THEN: reference_search should only be called once
assert media_item.plugin.manager.get_verses.call_count == 1
mocked_display_results.assert_called_once_with()
def text_reference_search(self, search_text, search_while_type=False):
"""
We are doing a 'Reference Search'.
This search is called on def text_search by Reference and Combined Searches.
"""
verse_refs = self.plugin.manager.parse_ref(self.bible.name, search_text)
self.search_results = self.reference_search(verse_refs, self.bible)
if self.second_bible and self.search_results:
self.second_search_results = self.reference_search(verse_refs, self.second_bible)
self.display_results()
def test_text_reference_search_dual_bible_no_results(media_item):
"""
Test text_reference_search when two bible are selected, but the search of the first bible does not return any
results
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked plugin.manager.get_verses
# WHEN: Calling text_reference_search with two bibles selected, but no results are found in the first bible
with patch.object(media_item, 'display_results') as mocked_display_results:
media_item.plugin.manager.get_verses.return_value = []
media_item.bible = mocked_bible_1
media_item.second_bible = mocked_bible_2
media_item.text_reference_search('Search Text')
# THEN: reference_search should only be called once
assert media_item.plugin.manager.get_verses.call_count == 1
mocked_display_results.assert_called_once_with()
def test_text_reference_search_dual_bible(media_item):
"""
Test text_reference_search when two bible are selected
"""
# GIVEN: An instance of :class:`MediaManagerItem` and mocked plugin.manager.get_verses
with patch.object(media_item, 'display_results') as mocked_display_results:
media_item.bible = mocked_bible_1
media_item.second_bible = mocked_bible_2
media_item.search_results = ''
media_item.second_search_results = ''
media_item.plugin.manager.get_verses.side_effect = lambda x, y, z: x
# WHEN: Calling text_reference_search with two bibles selected
media_item.text_reference_search('Search Text')
# THEN: reference_search should be called twice
assert media_item.plugin.manager.get_verses.call_count == 2
assert media_item.search_results == 'Bible 1'
assert media_item.second_search_results == 'Bible 2'
mocked_display_results.assert_called_once_with()
def test_on_text_search_single_bible(media_item):
"""
Test on_text_search when only one bible is selected
"""
# GIVEN: An instance of :class:`MediaManagerItem`
media_item.bible = mocked_bible_1
media_item.second_bible = None
# WHEN: Calling on_text_search and plugin.manager.verse_search returns a list of results
media_item.plugin.manager.verse_search.return_value = ['results', 'list']
with patch.object(media_item, 'display_results') as mocked_display_results:
media_item.on_text_search('Search Text')
# THEN: The search results should be the same as those returned by plugin.manager.verse_search
assert media_item.search_results == ['results', 'list']
mocked_display_results.assert_called_once_with()
def test_on_text_search_no_results(media_item):
"""
Test on_text_search when the search of the first bible does not return any results
"""
# GIVEN: An instance of :class:`MediaManagerItem`
media_item.bible = mocked_bible_1
media_item.second_bible = mocked_bible_2
# WHEN: Calling on_text_search and plugin.manager.verse_search returns an empty list
media_item.plugin.manager.verse_search.return_value = []
with patch.object(media_item, 'display_results') as mocked_display_results:
media_item.on_text_search('Search Text')
# THEN: The search results should be an empty list
assert media_item.search_results == []
mocked_display_results.assert_called_once_with()
def test_on_text_search_all_results_in_both_books(media_item, mocked_log):
"""
Test on_text_search when all of the results from the first bible are found in the second
"""
# GIVEN: An instance of :class:`MediaManagerItem` and some test data
mocked_verse_1 = MagicMock(**{'book.book_reference_id': 1, 'chapter': 2, 'verse': 3})
mocked_verse_1a = MagicMock(**{'book.book_reference_id': 1, 'chapter': 2, 'verse': 3})
mocked_verse_2 = MagicMock(**{'book.book_reference_id': 4, 'chapter': 5, 'verse': 6})
mocked_verse_2a = MagicMock(**{'book.book_reference_id': 4, 'chapter': 5, 'verse': 6})
media_item.bible = mocked_bible_1
media_item.second_bible = mocked_bible_2
media_item.second_search_results = []
# WHEN: Calling on_text_search and plugin.manager.verse_search returns a list of search results
media_item.plugin.manager.verse_search.return_value = [mocked_verse_1, mocked_verse_2]
media_item.second_bible.get_verses.side_effect = [[mocked_verse_1a], [mocked_verse_2a]]
with patch.object(media_item, 'display_results') as mocked_display_results:
media_item.on_text_search('Search Text')
# THEN: The search results for both bibles should be returned
assert media_item.search_results == [mocked_verse_1, mocked_verse_2]
assert media_item.second_search_results == [mocked_verse_1a, mocked_verse_2a]
assert mocked_log.debug.called is False
assert Registry().get('main_window').information_message.called is False
mocked_display_results.assert_called_once_with()
def test_on_text_search_not_all_results_in_both_books(media_item, mocked_log):
"""
Test on_text_search when not all of the results from the first bible are found in the second
"""
# GIVEN: An instance of :class:`MediaManagerItem` and some test data
mocked_verse_1 = MagicMock(**{'book.book_reference_id': 1, 'chapter': 2, 'verse': 3})
mocked_verse_1a = MagicMock(**{'book.book_reference_id': 1, 'chapter': 2, 'verse': 3})
mocked_verse_2 = MagicMock(**{'book.book_reference_id': 4, 'chapter': 5, 'verse': 6})
mocked_verse_3 = MagicMock(**{'book.book_reference_id': 7, 'chapter': 8, 'verse': 9})
media_item.bible = mocked_bible_1
media_item.second_bible = mocked_bible_2
media_item.second_search_results = []
# WHEN: Calling on_text_search and not all results are found in the second bible
media_item.plugin.manager.verse_search.return_value = [mocked_verse_1, mocked_verse_2, mocked_verse_3]
media_item.second_bible.get_verses.side_effect = [[mocked_verse_1a], [], []]
with patch.object(media_item, 'display_results') as mocked_display_results:
media_item.on_text_search('Search Text')
# THEN: The search results included in both bibles should be returned and the user should be notified of
# the missing verses
assert media_item.search_results == [mocked_verse_1]
assert media_item.second_search_results == [mocked_verse_1a]
assert mocked_log.debug.call_count == 2
assert Registry().get('main_window').information_message.called is True
mocked_display_results.assert_called_once_with()
def test_on_search_edit_text_changed_search_while_typing_disabled(media_item, mocked_timer):
"""
Test on_search_edit_text_changed when 'search while typing' is disabled
"""
# GIVEN: An instance of BibleMediaItem and mocked Settings which returns False when the value
# 'bibles/is search while typing enabled' is requested
Registry().get('settings').setValue('bibles/is search while typing enabled', False)
media_item.search_timer = mocked_timer
mocked_timer.isActive.return_value = False
# WHEN: Calling on_search_edit_text_changed
media_item.on_search_edit_text_changed()
# THEN: The method should not have checked if the timer is active
assert media_item.search_timer.isActive.called is False
def test_on_search_edit_text_changed_search_while_typing_enabled(media_item, mocked_timer):
"""
Test on_search_edit_text_changed when 'search while typing' is enabled
"""
# GIVEN: An instance of BibleMediaItem and mocked Settings which returns True when the value
# 'bibles/is search while typing enabled' is requested
Registry().get('settings').setValue('bibles/is search while typing enabled', True)
media_item.search_timer = mocked_timer
media_item.search_timer.isActive.return_value = False
media_item.bible = mocked_bible_1
media_item.bible.is_web_bible = False
# WHEN: Calling on_search_edit_text_changed
media_item.on_search_edit_text_changed()
# THEN: The method should start the search_timer
media_item.search_timer.isActive.assert_called_once_with()
media_item.search_timer.start.assert_called_once_with()
def test_on_search_timer_timeout(media_item):
"""
Test on_search_timer_timeout
"""
# GIVEN: An instance of BibleMediaItem
with patch.object(media_item, 'text_search') as mocked_text_search:
# WHEN: Calling on_search_timer_timeout
media_item.on_search_timer_timeout()
# THEN: The search_status should be set to SearchAsYouType and text_search should have been called
assert media_item.search_status == SearchStatus.SearchAsYouType
mocked_text_search.assert_called_once_with()
def test_display_results_no_results(media_item):
"""
Test the display_results method when there are no items to display
"""
# GIVEN: An instance of BibleMediaItem and a mocked build_display_results which returns an empty list
media_item.list_view = MagicMock()
media_item.bible = mocked_bible_1
media_item.second_bible = mocked_bible_2
media_item.search_results = []
with patch.object(media_item, 'build_display_results', return_value=[]):
# WHEN: Calling display_results with True
media_item.display_results()
# THEN: No items should be added to the list
assert media_item.list_view.addItem.called is False
def test_display_results_results(media_item):
"""
Test the display_results method when there are items to display
"""
# GIVEN: An instance of BibleMediaItem and a mocked build_display_results which returns a list of results
with patch.object(media_item, 'build_display_results', return_value=[
{'item_title': 'Title 1'}, {'item_title': 'Title 2'}]), \
patch.object(media_item, 'add_built_results_to_list_widget') as \
mocked_add_built_results_to_list_widget:
media_item.search_results = ['results']
media_item.list_view = MagicMock()
# WHEN: Calling display_results
media_item.display_results()
# THEN: addItem should have been with the display items
mocked_add_built_results_to_list_widget.assert_called_once_with(
[{'item_title': 'Title 1'}, {'item_title': 'Title 2'}])
def test_search_options(media_item):
"""
Test that the bible search options are returned
"""
# GIVEN: An instance of BibleMediaItem, and mocked primary bible setting
media_item.plugin.manager.get_bibles.return_value = MagicMock(keys=MagicMock(return_value=['a', 'b', 'c']))
media_item.settings.value = lambda key: {'bibles/primary bible': 'b'}[key]
# WHEN: calling search_options
result = media_item.search_options()
# THEN: result should be correct
assert result == [{'name': 'primary bible', 'list': ['a', 'b', 'c'], 'selected': 'b'}]
def test_set_search_option(media_item):
"""
Test that the bible set search option function works
"""
# GIVEN: An instance of BibleMediaItem, with a mocked get_bibles and a mocked populate_bible_combo_boxes function
media_item.plugin.manager.get_bibles.return_value = MagicMock(keys=MagicMock(return_value=['a', 'b', 'c']))
populate_bible_combo_boxes = MagicMock()
Registry().remove_function('populate_bible_combo_boxes', media_item.populate_bible_combo_boxes)
Registry().register_function('populate_bible_combo_boxes', populate_bible_combo_boxes)
# WHEN: calling set_search_option
result = media_item.set_search_option('primary bible', 'c')
# THEN: result should be correct and the new bible set, and combo boxes updated
assert result is True
media_item.settings.setValue.assert_called_once_with('bibles/primary bible', 'c')
populate_bible_combo_boxes.assert_called_once()
def test_set_search_option_invalid_option(media_item):
"""
Test that the bible set search option function rejects when using non existing option
"""
# GIVEN: An instance of BibleMediaItem, with a mocked get_bibles and a mocked populate_bible_combo_boxes function
media_item.plugin.manager.get_bibles.return_value = MagicMock(keys=MagicMock(return_value=['a', 'b', 'c']))
populate_bible_combo_boxes = MagicMock()
Registry().remove_function('populate_bible_combo_boxes', media_item.populate_bible_combo_boxes)
Registry().register_function('populate_bible_combo_boxes', populate_bible_combo_boxes)
# WHEN: calling set_search_option with invalid option
result = media_item.set_search_option('not an option', 'a')
# THEN: result should be False and nothing set, and combo boxes not updated
assert result is False
media_item.settings.setValue.assert_not_called()
populate_bible_combo_boxes.assert_not_called()
def test_set_search_option_invalid_value(media_item):
"""
Test that the bible set search option function rejects when using non existing value
"""
# GIVEN: An instance of BibleMediaItem, with a mocked get_bibles and a mocked populate_bible_combo_boxes function
media_item.plugin.manager.get_bibles.return_value = MagicMock(keys=MagicMock(return_value=['a', 'b', 'c']))
populate_bible_combo_boxes = MagicMock()
Registry().remove_function('populate_bible_combo_boxes', media_item.populate_bible_combo_boxes)
Registry().register_function('populate_bible_combo_boxes', populate_bible_combo_boxes)
# WHEN: calling set_search_option with invalid value
result = media_item.set_search_option('primary bible', 'not a value')
# THEN: result should be False and nothing set, and combo boxes not updated
assert result is False
media_item.settings.setValue.assert_not_called()
populate_bible_combo_boxes.assert_not_called()
def test_generate_slide_data_data_string(media_item):
"""
Test that the generated_slide_data provides data values for the api
"""
# GIVEN: Mocked service item and some pretend bible slide data
mocked_service_item = MagicMock()
slide_data = {
'book': 'Matthew',
'chapter': '1',
'verse': '2',
'version': 'Bible version 104',
'copyright': 'copywrong',
'permissions': 'all the permissions',
'second_bible': 'Second bible thing',
'second_copyright': 'copywrite',
'second_permissions': 'no use allowed',
'second_version': 'Bible version 4567',
'text': 'text from matthew 1:2',
'second_text': 'different text from matthew 1:2'
}
mocked_items = [MagicMock(**{'data.return_value': slide_data})]
media_item.format_verse = MagicMock()
media_item.settings_tab = MagicMock()
# WHEN: Called generate_slide_data
media_item.generate_slide_data(mocked_service_item, item=mocked_items)
# THEN: both primary and secondary bible info should be included in the returned service item data string
assert mocked_service_item.data_string == {
'bibles': [
{
'version': 'Bible version 104',
'copyright': 'copywrong',
'permissions': 'all the permissions'
},
{
'version': 'Bible version 4567',
'copyright': 'copywrite',
'permissions': 'no use allowed'
}
]
}
def test_generate_slide_data_data_string_one_bible(media_item):
"""
Test that the generated_slide_data provides data values for the api
"""
# GIVEN: Mocked service item and some pretend bible slide data
mocked_service_item = MagicMock()
slide_data = {
'book': 'Matthew',
'chapter': '1',
'verse': '2',
'version': 'Bible version 104',
'copyright': 'copywrong',
'permissions': 'all the permissions',
'second_bible': '',
'text': 'text from matthew 1:2'
}
mocked_items = [MagicMock(**{'data.return_value': slide_data})]
media_item.format_verse = MagicMock()
media_item.settings_tab = MagicMock()
# WHEN: Called generate_slide_data
media_item.generate_slide_data(mocked_service_item, item=mocked_items)
# THEN: both primary and secondary bible info should be included in the returned service item data string
assert mocked_service_item.data_string == {
'bibles': [
{
'version': 'Bible version 104',
'copyright': 'copywrong',
'permissions': 'all the permissions'
}
]
}