~launchpad-pqm/launchpad/devel

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
# Copyright 2009 Canonical Ltd.  This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).

"""IFAQCollection browser views."""

__metaclass__ = type

__all__ = [
    'FAQCollectionMenu',
    'SearchFAQsView',
    ]

from urllib import urlencode

from canonical.launchpad import _
from canonical.launchpad.webapp import (
    canonical_url,
    Link,
    NavigationMenu,
    )
from canonical.launchpad.webapp.batching import BatchNavigator
from canonical.launchpad.webapp.menu import enabled_with_permission
from lp.answers.enums import (
    QUESTION_STATUS_DEFAULT_SEARCH,
    QuestionSort,
    )
from lp.answers.interfaces.faqcollection import (
    FAQSort,
    IFAQCollection,
    ISearchFAQsForm,
    )
from lp.app.browser.launchpadform import (
    action,
    LaunchpadFormView,
    safe_action,
    )
from lp.registry.interfaces.projectgroup import IProjectGroup
from lp.services.propertycache import cachedproperty


class FAQCollectionMenu(NavigationMenu):
    """Base menu definition for `IFAQCollection`."""

    usedfor = IFAQCollection
    facet = 'answers'
    links = ['list_all', 'create_faq']

    def list_all(self):
        """Return a Link to list all FAQs."""
        # We adapt to IFAQCollection so that the link can be used
        # on objects which don't provide `IFAQCollection` directly, but for
        # which an adapter exists that gives the proper context.
        collection = IFAQCollection(self.context)
        url = canonical_url(collection, rootsite='answers') + '/+faqs'
        return Link(url, 'All FAQs', icon='info')

    @enabled_with_permission('launchpad.Append')
    def create_faq(self):
        """Return a Link to create a new FAQ."""
        collection = IFAQCollection(self.context)
        if IProjectGroup.providedBy(self.context):
            url = ''
            enabled = False
        else:
            url = canonical_url(
                collection, view_name='+createfaq', rootsite='answers')
            enabled = True
        return Link(url, 'Create a new FAQ', icon='add', enabled=enabled)


class SearchFAQsView(LaunchpadFormView):
    """View to list and search FAQs."""

    schema = ISearchFAQsForm

    # This attribute contains the search_text to use.
    search_text = None

    # This attribute is updated to the number of matching questions when
    # the user does a search.
    matching_questions_count = 0

    @property
    def page_title(self):
        """Return the page_title that should be used for the listing."""
        replacements = dict(
            displayname=self.context.displayname,
            search_text=self.search_text)
        if self.search_text:
            return _(u'FAQs matching \u201c${search_text}\u201d for '
                     u'$displayname', mapping=replacements)
        else:
            return _('FAQs for $displayname', mapping=replacements)

    label = page_title

    @property
    def empty_listing_message(self):
        """Return the message to render when there are no FAQs to display."""
        replacements = dict(
            displayname=self.context.displayname,
            search_text=self.search_text)
        if self.search_text:
            return _(u'There are no FAQs for $displayname matching '
                     u'\u201c${search_text}\u201d.', mapping=replacements)
        else:
            return _('There are no FAQs for $displayname.',
                     mapping=replacements)

    def getMatchingFAQs(self):
        """Return a BatchNavigator of the matching FAQs."""
        faqs = self.context.searchFAQs(search_text=self.search_text)
        return BatchNavigator(faqs, self.request)

    @property
    def portlet_action(self):
        """The action URL of the portlet form."""
        return canonical_url(
            self.context, view_name='+faqs', rootsite='answers')

    @cachedproperty
    def latest_faqs(self):
        """Return the latest faqs created for this target.

        This is used by the +portlet-listfaqs view.
        """
        quantity = 5
        faqs = self.context.searchFAQs(
            search_text=self.search_text, sort=FAQSort.NEWEST_FIRST)
        return faqs[:quantity]

    @safe_action
    @action(_('Search'), name='search')
    def search_action(self, action, data):
        """Filter the search results by keywords."""
        self.search_text = data.get('search_text', None)
        if self.search_text:
            matching_questions = self.context.searchQuestions(
                search_text=self.search_text)
            self.matching_questions_count = matching_questions.count()

    @property
    def matching_questions_url(self):
        """Return the URL to the questions matching the same keywords."""
        return canonical_url(self.context) + '/+questions?' + urlencode(
            {'field.status': [
                status.title for status in QUESTION_STATUS_DEFAULT_SEARCH],
             'field.search_text': self.search_text,
             'field.actions.search': 'Search',
             'field.sort': QuestionSort.RELEVANCY.title,
             'field.language-empty-marker': 1}, doseq=True)