diff --git a/gajim/chat_control_base.py b/gajim/chat_control_base.py
index bb45a5b70f9552f658c66873bf6b6ffee73aebd1..8cf4b64a7e0c3b44cd03a4a33c3de8825e1a1e20 100644
--- a/gajim/chat_control_base.py
+++ b/gajim/chat_control_base.py
@@ -174,6 +174,7 @@ def __init__(self, parent_win, widget_name, jid, acct,
 
         self._scrolled_view = ScrolledView()
         self._scrolled_view.add(self.conversation_view)
+        self._scrolled_view.set_focus_vadjustment(Gtk.Adjustment())
         self.xml.textview_box.add(self._scrolled_view)
         self.xml.textview_box.reorder_child(self._scrolled_view, 2)
         self._scrolled_view.connect('request-history',
@@ -1143,6 +1144,7 @@ def add_message(self,
             display_marking=displaymarking,
             message_id=message_id,
             correct_id=correct_id,
+            log_line_id=msg_log_id,
             additional_data=additional_data,
             marker=marker,
             error=error)
@@ -1390,6 +1392,25 @@ def set_control_active(self, state):
     def scroll_to_end(self, force=False):
         self.conversation_view.scroll_to_end(force)
 
+    def scroll_to_message(self, log_line_id, timestamp):
+        row = self.conversation_view.get_row_by_log_line_id(log_line_id)
+        if row is None:
+            first_row = self.conversation_view.get_first_message_row()
+            if first_row is None:
+                first_timestamp = time.time()
+            else:
+                first_timestamp = first_row.db_timestamp
+            messages = app.storage.archive.get_conversation_between(
+                self.account, self.contact.jid, first_timestamp, timestamp)
+            if not messages:
+                return
+
+            self.add_messages(messages)
+
+        GLib.idle_add(
+            self.conversation_view.scroll_to_message_and_highlight,
+            log_line_id)
+
     def fetch_n_lines_history(self, _scrolled, n_lines):
         row = self.conversation_view.get_first_message_row()
         if row is None:
@@ -1414,6 +1435,9 @@ def fetch_n_lines_history(self, _scrolled, n_lines):
             self._scrolled_view.set_history_complete(True)
             return
 
+        self.add_messages(messages)
+
+    def add_messages(self, messages):
         for msg in messages:
             if not msg:
                 continue
@@ -1439,6 +1463,7 @@ def fetch_n_lines_history(self, _scrolled, n_lines):
                 subject=msg.subject,
                 additional_data=msg.additional_data,
                 message_id=msg.message_id,
+                log_line_id=msg.log_line_id,
                 marker=msg.marker,
                 error=msg.error,
                 history=True)
diff --git a/gajim/common/storage/archive.py b/gajim/common/storage/archive.py
index f9a0bae50b34e211c226cfd626bd4d38872f9511..c6a233df00e9041e394a5cd7c6467d806a7f5c7b 100644
--- a/gajim/common/storage/archive.py
+++ b/gajim/common/storage/archive.py
@@ -208,6 +208,9 @@ def _get_jid_ids_from_db(self):
             self._jid_ids[row.jid] = row
             self._jid_ids_reversed[row.jid_id] = row
 
+    def get_jid_from_id(self, jid_id):
+        return self._jid_ids_reversed[jid_id]
+
     def get_jids_in_db(self):
         return self._jid_ids.keys()
 
@@ -418,6 +421,40 @@ def get_last_conversation_line(self, account, jid):
 
         return self._con.execute(sql, tuple(jids)).fetchone()
 
+    @timeit
+    def get_conversation_between(self, account, jid, before, after):
+        """
+        Load all lines of conversation with jid between two timestamps
+
+        :param account:         The account
+
+        :param jid:             The jid for which we request the conversation
+
+        :param before:          latest timestamp
+
+        :param after:           earliest timestamp
+
+        returns a list of namedtuples
+        """
+        jids = self._get_family_jids(account, jid)
+        account_id = self.get_account_id(account)
+
+        sql = '''
+            SELECT contact_name, time, kind, show, message, subject,
+                   additional_data, log_line_id, message_id,
+                   error as "error [common_error]",
+                   marker as "marker [marker]"
+            FROM logs NATURAL JOIN jids WHERE jid IN ({jids})
+            AND account_id = {account_id}
+            AND time < ? AND time >= ?
+            ORDER BY time DESC, log_line_id DESC
+            '''.format(jids=', '.join('?' * len(jids)),
+                       account_id=account_id)
+
+        return self._con.execute(
+            sql,
+            tuple(jids) + (before, after)).fetchall()
+
     @timeit
     def get_messages_for_date(self, account, jid, date):
         """
@@ -477,6 +514,9 @@ def search_log(self, account, jid, query, date=None):
         """
         jids = self._get_family_jids(account, jid)
 
+        kinds = map(str, [KindConstant.STATUS,
+                          KindConstant.GCSTATUS])
+
         if date:
             delta = datetime.timedelta(
                 hours=23, minutes=59, seconds=59, microseconds=999999)
@@ -491,12 +531,36 @@ def search_log(self, account, jid, query, date=None):
                additional_data, log_line_id
         FROM logs NATURAL JOIN jids WHERE jid IN ({jids})
         AND message LIKE like(?) {date_search}
+        AND kind NOT IN ({kinds})
         ORDER BY time DESC, log_line_id
         '''.format(jids=', '.join('?' * len(jids)),
-                   date_search=between if date else '')
+                   date_search=between if date else '',
+                   kinds=', '.join(kinds))
 
         return self._con.execute(sql, tuple(jids) + (query,)).fetchall()
 
+    @timeit
+    def search_all_logs(self, query):
+        """
+        Search all conversation logs for messages containing the `query`
+        string.
+
+        :param query:   A search string
+
+        returns a list of namedtuples
+        """
+        kinds = map(str, [KindConstant.STATUS,
+                          KindConstant.GCSTATUS])
+        sql = '''
+        SELECT account_id, jid_id, contact_name, time, kind, show, message,
+        subject, additional_data, log_line_id
+        FROM logs WHERE message LIKE like(?)
+        AND kind NOT IN ({kinds})
+        ORDER BY time DESC, log_line_id
+        '''.format(kinds=', '.join(kinds))
+
+        return self._con.execute(sql, (query,)).fetchall()
+
     @timeit
     def get_days_with_logs(self, account, jid, year, month):
         """
diff --git a/gajim/data/gui/main.ui b/gajim/data/gui/main.ui
index 75da29e866fc95073b84b1764182dc4fccc8a6f0..32891fb471a521f2b9c41db9523deae166f124b5 100644
--- a/gajim/data/gui/main.ui
+++ b/gajim/data/gui/main.ui
@@ -141,7 +141,7 @@
             <property name="position">250</property>
             <property name="position-set">True</property>
             <child>
-              <!-- n-columns=3 n-rows=3 -->
+              <!-- n-columns=1 n-rows=2 -->
               <object class="GtkGrid" id="middle_grid">
                 <property name="visible">True</property>
                 <property name="can-focus">False</property>
@@ -287,27 +287,6 @@
                     <property name="top-attach">1</property>
                   </packing>
                 </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
                 <style>
                   <class name="middle-grid"/>
                 </style>
@@ -318,38 +297,26 @@
               </packing>
             </child>
             <child>
-              <!-- n-columns=3 n-rows=3 -->
+              <!-- n-columns=1 n-rows=1 -->
               <object class="GtkGrid" id="right_grid">
                 <property name="visible">True</property>
                 <property name="can-focus">False</property>
                 <property name="hexpand">True</property>
                 <property name="vexpand">True</property>
                 <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
-                </child>
-                <child>
-                  <placeholder/>
+                  <object class="GtkOverlay" id="right_grid_overlay">
+                    <property name="visible">True</property>
+                    <property name="can-focus">False</property>
+                    <property name="hexpand">True</property>
+                    <property name="vexpand">True</property>
+                    <child>
+                      <placeholder/>
+                    </child>
+                  </object>
+                  <packing>
+                    <property name="left-attach">0</property>
+                    <property name="top-attach">0</property>
+                  </packing>
                 </child>
               </object>
               <packing>
diff --git a/gajim/data/gui/search_view.ui b/gajim/data/gui/search_view.ui
new file mode 100644
index 0000000000000000000000000000000000000000..ed1dc4ad975a5625a52e79595552f879629b99ef
--- /dev/null
+++ b/gajim/data/gui/search_view.ui
@@ -0,0 +1,289 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!-- Generated with glade 3.38.2 -->
+<interface>
+  <requires lib="gtk+" version="3.24"/>
+  <object class="GtkBox" id="header_box">
+    <property name="visible">True</property>
+    <property name="can-focus">False</property>
+    <property name="hexpand">True</property>
+    <property name="spacing">6</property>
+    <child>
+      <object class="GtkSeparator">
+        <property name="visible">True</property>
+        <property name="can-focus">False</property>
+        <property name="valign">center</property>
+      </object>
+      <packing>
+        <property name="expand">True</property>
+        <property name="fill">True</property>
+        <property name="position">0</property>
+      </packing>
+    </child>
+    <child>
+      <object class="GtkLabel" id="header_name_label">
+        <property name="visible">True</property>
+        <property name="can-focus">False</property>
+        <property name="ellipsize">end</property>
+        <property name="max-width-chars">15</property>
+      </object>
+      <packing>
+        <property name="expand">False</property>
+        <property name="fill">True</property>
+        <property name="position">1</property>
+      </packing>
+    </child>
+    <child>
+      <object class="GtkLabel" id="header_date_label">
+        <property name="visible">True</property>
+        <property name="can-focus">False</property>
+      </object>
+      <packing>
+        <property name="expand">False</property>
+        <property name="fill">True</property>
+        <property name="position">2</property>
+      </packing>
+    </child>
+    <child>
+      <object class="GtkSeparator">
+        <property name="visible">True</property>
+        <property name="can-focus">False</property>
+        <property name="valign">center</property>
+      </object>
+      <packing>
+        <property name="expand">True</property>
+        <property name="fill">True</property>
+        <property name="position">3</property>
+      </packing>
+    </child>
+    <style>
+      <class name="search-view-header"/>
+    </style>
+  </object>
+  <!-- n-columns=3 n-rows=2 -->
+  <object class="GtkGrid" id="result_row_grid">
+    <property name="visible">True</property>
+    <property name="can-focus">False</property>
+    <property name="row-spacing">3</property>
+    <property name="column-spacing">12</property>
+    <child>
+      <object class="GtkImage" id="row_avatar">
+        <property name="visible">True</property>
+        <property name="can-focus">False</property>
+        <property name="valign">start</property>
+      </object>
+      <packing>
+        <property name="left-attach">0</property>
+        <property name="top-attach">0</property>
+        <property name="height">2</property>
+      </packing>
+    </child>
+    <child>
+      <object class="GtkLabel" id="row_time_label">
+        <property name="visible">True</property>
+        <property name="can-focus">False</property>
+        <property name="halign">end</property>
+        <property name="single-line-mode">True</property>
+        <style>
+          <class name="dim-label"/>
+        </style>
+      </object>
+      <packing>
+        <property name="left-attach">2</property>
+        <property name="top-attach">0</property>
+      </packing>
+    </child>
+    <child>
+      <object class="GtkLabel" id="row_name_label">
+        <property name="visible">True</property>
+        <property name="can-focus">False</property>
+        <property name="halign">start</property>
+        <property name="ellipsize">end</property>
+        <property name="single-line-mode">True</property>
+        <style>
+          <class name="bold"/>
+        </style>
+      </object>
+      <packing>
+        <property name="left-attach">1</property>
+        <property name="top-attach">0</property>
+      </packing>
+    </child>
+    <child>
+      <placeholder/>
+    </child>
+    <child>
+      <placeholder/>
+    </child>
+    <style>
+      <class name="search-view-row-grid"/>
+    </style>
+  </object>
+  <object class="GtkBox" id="search_box">
+    <property name="visible">True</property>
+    <property name="can-focus">False</property>
+    <property name="hexpand">True</property>
+    <property name="orientation">vertical</property>
+    <property name="spacing">6</property>
+    <child>
+      <object class="GtkBox">
+        <property name="visible">True</property>
+        <property name="can-focus">False</property>
+        <property name="spacing">12</property>
+        <child type="center">
+          <object class="GtkLabel">
+            <property name="visible">True</property>
+            <property name="can-focus">False</property>
+            <property name="label" translatable="yes">Search</property>
+            <style>
+              <class name="bold16"/>
+            </style>
+          </object>
+          <packing>
+            <property name="expand">False</property>
+            <property name="fill">True</property>
+            <property name="position">2</property>
+          </packing>
+        </child>
+        <child>
+          <object class="GtkButton">
+            <property name="visible">True</property>
+            <property name="can-focus">True</property>
+            <property name="receives-default">True</property>
+            <property name="tooltip-text" translatable="yes">Close Search</property>
+            <property name="relief">none</property>
+            <signal name="clicked" handler="_on_hide_clicked" swapped="no"/>
+            <child>
+              <object class="GtkImage">
+                <property name="visible">True</property>
+                <property name="can-focus">False</property>
+                <property name="icon-name">window-close-symbolic</property>
+              </object>
+            </child>
+          </object>
+          <packing>
+            <property name="expand">False</property>
+            <property name="fill">True</property>
+            <property name="position">0</property>
+          </packing>
+        </child>
+      </object>
+      <packing>
+        <property name="expand">False</property>
+        <property name="fill">True</property>
+        <property name="position">0</property>
+      </packing>
+    </child>
+    <child>
+      <object class="GtkSearchEntry" id="search_entry">
+        <property name="width-request">200</property>
+        <property name="visible">True</property>
+        <property name="can-focus">True</property>
+        <property name="halign">center</property>
+        <property name="primary-icon-name">edit-find-symbolic</property>
+        <property name="primary-icon-activatable">False</property>
+        <property name="primary-icon-sensitive">False</property>
+        <signal name="activate" handler="_on_search" swapped="no"/>
+      </object>
+      <packing>
+        <property name="expand">False</property>
+        <property name="fill">True</property>
+        <property name="position">1</property>
+      </packing>
+    </child>
+    <child>
+      <object class="GtkCheckButton" id="search_checkbutton">
+        <property name="label" translatable="yes">Search everywhere</property>
+        <property name="visible">True</property>
+        <property name="can-focus">True</property>
+        <property name="receives-default">False</property>
+        <property name="halign">center</property>
+        <property name="active">True</property>
+        <property name="draw-indicator">True</property>
+      </object>
+      <packing>
+        <property name="expand">False</property>
+        <property name="fill">True</property>
+        <property name="position">2</property>
+      </packing>
+    </child>
+    <child>
+      <object class="GtkSeparator">
+        <property name="visible">True</property>
+        <property name="can-focus">False</property>
+      </object>
+      <packing>
+        <property name="expand">False</property>
+        <property name="fill">True</property>
+        <property name="position">3</property>
+      </packing>
+    </child>
+    <child>
+      <object class="GtkScrolledWindow">
+        <property name="visible">True</property>
+        <property name="can-focus">True</property>
+        <child>
+          <object class="GtkViewport">
+            <property name="visible">True</property>
+            <property name="can-focus">False</property>
+            <child>
+              <object class="GtkListBox" id="results_listbox">
+                <property name="visible">True</property>
+                <property name="can-focus">False</property>
+                <property name="selection-mode">none</property>
+                <signal name="row-activated" handler="_on_row_activated" swapped="no"/>
+                <child type="placeholder">
+                  <object class="GtkBox" id="placeholder">
+                    <property name="visible">True</property>
+                    <property name="can-focus">False</property>
+                    <property name="valign">center</property>
+                    <property name="orientation">vertical</property>
+                    <property name="spacing">12</property>
+                    <child>
+                      <object class="GtkImage">
+                        <property name="visible">True</property>
+                        <property name="can-focus">False</property>
+                        <property name="icon-name">system-search-symbolic</property>
+                        <property name="icon_size">6</property>
+                        <style>
+                          <class name="dim-label"/>
+                        </style>
+                      </object>
+                      <packing>
+                        <property name="expand">False</property>
+                        <property name="fill">True</property>
+                        <property name="position">0</property>
+                      </packing>
+                    </child>
+                    <child>
+                      <object class="GtkLabel">
+                        <property name="visible">True</property>
+                        <property name="can-focus">False</property>
+                        <property name="label" translatable="yes">No results</property>
+                        <style>
+                          <class name="dim-label"/>
+                        </style>
+                      </object>
+                      <packing>
+                        <property name="expand">False</property>
+                        <property name="fill">True</property>
+                        <property name="position">1</property>
+                      </packing>
+                    </child>
+                  </object>
+                </child>
+              </object>
+            </child>
+          </object>
+        </child>
+      </object>
+      <packing>
+        <property name="expand">True</property>
+        <property name="fill">True</property>
+        <property name="position">4</property>
+      </packing>
+    </child>
+    <style>
+      <class name="search-view"/>
+    </style>
+  </object>
+</interface>
diff --git a/gajim/data/style/gajim.css b/gajim/data/style/gajim.css
index 95def94cfddce05d0f507ff88eba003c0fdab399..c9f982e748e7a70d5f1818dfb249521186f2493a 100644
--- a/gajim/data/style/gajim.css
+++ b/gajim/data/style/gajim.css
@@ -59,8 +59,15 @@ .conversation-row grid textview text {
 .conversation-mention-highlight {
     background-color: rgb(255, 215, 194);
 }
+@keyframes highlight {
+    from { background: rgb(194, 215, 255) }
+    to { background: transparent }
+}
 .conversation-search-highlight {
-    background-color: rgb(194, 215, 255);
+    animation-duration: 3s;
+    animation-timing-function: ease-out;
+    animation-iteration-count: 1;
+    animation-name: highlight;
 }
 .conversation-system-row {
     padding: 18px;
@@ -303,6 +310,33 @@ .header-box-first label {
 	padding: 0px 18px 6px 18px;
 }
 
+/* Seach view */
+.search-view {
+    padding: 12px;
+    border-left: 1px solid @borders;
+    background-color: @theme_base_color;
+}
+.search-view-row-grid textview {
+	background: transparent;
+}
+.search-view-row-grid textview text {
+	background: transparent;
+}
+.search-view-row-grid label {
+    font-size: small;
+}
+.search-view-counter label {
+    font-style: italic;
+    color: @insensitive_fg_color;
+}
+.search-view-header {
+    padding: 12px 6px 6px 6px;
+}
+.search-view-header label {
+    font-size: small;
+    color: @insensitive_fg_color;
+}
+
 /*  Profile window */
 #NicknameEntry:disabled {
     font-size: 28px;
diff --git a/gajim/gtk/conversation/view.py b/gajim/gtk/conversation/view.py
index 141729b783c9cc87d474ad17d015688e00595e4a..0898aeca4f4687393401d917fc814f0c9621bf78 100644
--- a/gajim/gtk/conversation/view.py
+++ b/gajim/gtk/conversation/view.py
@@ -316,6 +316,20 @@ def reduce_message_count(self):
 
         return successful
 
+    def scroll_to_message_and_highlight(self, log_line_id):
+        highlight_row = None
+        for row in self.get_children():
+            row.get_style_context().remove_class(
+                'conversation-search-highlight')
+            if row.log_line_id == log_line_id:
+                highlight_row = row
+
+        if highlight_row is not None:
+            highlight_row.get_style_context().add_class(
+                'conversation-search-highlight')
+            # This scrolls the ListBox to the highlighted row
+            highlight_row.grab_focus()
+
     def _get_row_by_message_id(self, id_):
         return self._message_id_row_map.get(id_)
 
diff --git a/gajim/gtk/main.py b/gajim/gtk/main.py
index f073d0cfa8dcf5687c39f87ba1cfcff4ea29da4e..300eb0dc5649e0e6d5643c5e5765af5c7cc32156 100644
--- a/gajim/gtk/main.py
+++ b/gajim/gtk/main.py
@@ -13,6 +13,7 @@
 
 from gajim.gui.account_page import AccountPage
 from gajim.gui.adhoc import AdHocCommand
+from gajim.gui.search_view import SearchView
 from gajim.gui.chat_list_stack import ChatListStack
 from gajim.gui.chat_stack import ChatStack
 from gajim.gui.account_side_bar import AccountSideBar
@@ -57,9 +58,20 @@ def __init__(self):
         self._ui.app_image.set_from_surface(surface)
 
         self._chat_stack = ChatStack()
-        self._ui.right_grid.add(self._chat_stack)
+        self._ui.right_grid_overlay.add(self._chat_stack)
+
+        self._search_view = SearchView()
+        self._search_view.connect('hide-search', self._on_search_hide)
+
+        self._search_revealer = Gtk.Revealer()
+        self._search_revealer.set_reveal_child(True)
+        self._search_revealer.set_halign(Gtk.Align.END)
+        self._search_revealer.set_no_show_all(True)
+        self._search_revealer.add(self._search_view)
+        self._ui.right_grid_overlay.add_overlay(self._search_revealer)
 
         self._chat_list_stack = ChatListStack(self, self._ui, self._chat_stack)
+        self._chat_list_stack.connect('chat-selected', self._on_chat_selected)
         self._ui.chat_list_scrolled.add(self._chat_list_stack)
 
         self._workspace_side_bar = WorkspaceSideBar(self._chat_list_stack)
@@ -138,6 +150,7 @@ def _add_actions(self):
             ('toggle-chat-pinned', 'as', self._toggle_chat_pinned),
             ('move-chat-to-workspace', 'as', self._move_chat_to_workspace),
             ('add-to-roster', 'as', self._add_to_roster),
+            ('search-history', None, self._on_search_history),
         ]
 
         for action in actions:
@@ -207,6 +220,10 @@ def _on_action(self, action, _param):
         if res != Gdk.EVENT_PROPAGATE:
             return res
 
+        if action == 'escape':
+            if self._search_revealer.get_reveal_child():
+                self._search_revealer.hide()
+
         # if action == 'escape' and app.settings.get('escape_key_closes'):
         #     self.remove_tab(control, self.CLOSE_ESC)
         #     return
@@ -559,6 +576,22 @@ def _load_chats(self):
     def _on_start_chat_clicked(_button):
         app.app.activate_action('start-chat', GLib.Variant('s', ''))
 
+    def _on_chat_selected(self, _chat_list_stack, _workspace_id, *args):
+        control = self.get_active_control()
+        if control is not None:
+            self._search_view.set_context(control.account, control.contact.jid)
+
+    def _on_search_history(self, _action, _param):
+        control = self.get_active_control()
+        if control is not None:
+            self._search_view.set_context(control.account, control.contact.jid)
+        self._search_view.clear()
+        self._search_revealer.show()
+        self._search_view.set_focus()
+
+    def _on_search_hide(self, *args):
+        self._search_revealer.hide()
+
     def _on_event(self, event):
         if event.name == 'caps-update':
             #TODO
diff --git a/gajim/gtk/search_view.py b/gajim/gtk/search_view.py
new file mode 100644
index 0000000000000000000000000000000000000000..f2b3ffe69cb329fc18e813ca5104330f9c572ec0
--- /dev/null
+++ b/gajim/gtk/search_view.py
@@ -0,0 +1,253 @@
+# This file is part of Gajim.
+#
+# Gajim 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; version 3 only.
+#
+# Gajim 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 Gajim. If not, see <http://www.gnu.org/licenses/>.
+
+import logging
+import time
+
+from gi.repository import Gdk
+from gi.repository import GObject
+from gi.repository import Gtk
+
+from gajim.common import app
+from gajim.common.const import AvatarSize
+from gajim.common.const import KindConstant
+from gajim.common.i18n import _
+from gajim.common.styling import process
+
+from .conversation.message_widget import MessageWidget
+from .util import get_builder
+
+log = logging.getLogger('gajim.gui.search_view')
+
+
+class SearchView(Gtk.Box):
+
+    __gsignals__ = {
+        'hide-search': (
+            GObject.SignalFlags.RUN_FIRST,
+            None,
+            ()),
+    }
+
+    def __init__(self):
+        Gtk.Box.__init__(self)
+        self.set_size_request(300, -1)
+
+        self._account = None
+        self._jid = None
+
+        self._ui = get_builder('search_view.ui')
+        self._ui.results_listbox.set_header_func(self._header_func)
+        self.add(self._ui.search_box)
+
+        self._ui.connect_signals(self)
+        self.connect('key-press-event', self._on_key_press)
+        self.show_all()
+
+    def _on_key_press(self, _widget, event):
+        if event.keyval == Gdk.KEY_Escape:
+            self.emit('hide-search')
+
+    @staticmethod
+    def _header_func(row, before):
+        if before is None:
+            if row.type == 'counter':
+                row.set_header(None)
+            else:
+                row.set_header(RowHeader(row.account, row.jid, row.time))
+        else:
+            date1 = time.strftime('%x', time.localtime(row.time))
+            date2 = time.strftime('%x', time.localtime(before.time))
+            if before.jid != row.jid:
+                row.set_header(RowHeader(row.account, row.jid, row.time))
+            elif date1 != date2:
+                row.set_header(RowHeader(row.account, row.jid, row.time))
+            else:
+                row.set_header(None)
+
+    def _on_hide_clicked(self, _button):
+        self.emit('hide-search')
+
+    def clear(self):
+        self._ui.search_entry.set_text('')
+        self._clear_results()
+
+    def _clear_results(self):
+        for row in self._ui.results_listbox.get_children():
+            self._ui.results_listbox.remove(row)
+
+    def _on_search(self, entry):
+        self._clear_results()
+        text = entry.get_text()
+        if text == '':
+            return
+
+        accounts = self._get_accounts()
+        everywhere = self._ui.search_checkbutton.get_active()
+        context = self._account is not None and self._jid is not None
+
+        if not context or everywhere:
+            # Global search
+            results = app.storage.archive.search_all_logs(text)
+            results_count = len(results)
+            if results_count:
+                self._ui.results_listbox.add(CounterRow(results_count))
+            for msg in results:
+                result_row = ResultRow(
+                    msg,
+                    accounts.get(msg.account_id),
+                    app.storage.archive.get_jid_from_id(msg.jid_id).jid)
+                self._ui.results_listbox.add(result_row)
+        else:
+            results = app.storage.archive.search_log(
+                self._account, self._jid, text)
+            results_count = len(results)
+            if results_count:
+                self._ui.results_listbox.add(CounterRow(results_count))
+            for msg in results:
+                result_row = ResultRow(msg, self._account, self._jid)
+                self._ui.results_listbox.add(result_row)
+
+    @staticmethod
+    def _get_accounts():
+        accounts = {}
+        for account in app.settings.get_accounts():
+            account_id = app.storage.archive.get_account_id(account)
+            accounts[account_id] = account
+        return accounts
+
+    @staticmethod
+    def _on_row_activated(_listbox, row):
+        if row.type == 'counter':
+            return
+
+        control = app.window.get_active_control()
+        if control is not None:
+            if control.contact.jid == row.jid:
+                control.scroll_to_message(row.log_line_id, row.timestamp)
+                return
+
+        # Wrong chat or no control opened
+        # TODO: type 'pm' is KindConstant.CHAT_MSG_RECV, too
+        app.window.add_chat(row.account, row.jid, row.type, select=True)
+        control = app.window.get_active_control()
+        control.scroll_to_message(row.log_line_id, row.timestamp)
+
+    def set_focus(self):
+        self._ui.search_entry.grab_focus()
+
+    def set_context(self, account, jid):
+        self._account = account
+        self._jid = jid
+        self._ui.search_checkbutton.set_active(False)
+
+
+class RowHeader(Gtk.Box):
+    def __init__(self, account, jid, timestamp):
+        Gtk.Box.__init__(self)
+        self.set_hexpand(True)
+
+        self._ui = get_builder('search_view.ui')
+        self.add(self._ui.header_box)
+
+        client = app.get_client(account)
+        contact = client.get_module('Contacts').get_contact(jid)
+        self._ui.header_name_label.set_text(contact.name or '')
+
+        local_time = time.localtime(timestamp)
+        date = time.strftime('%x', local_time)
+        self._ui.header_date_label.set_text(date)
+
+        self.show_all()
+
+
+class CounterRow(Gtk.ListBoxRow):
+    def __init__(self, count):
+        Gtk.ListBoxRow.__init__(self)
+        self.type = 'counter'
+        self.jid = ''
+        self.time = 0
+        self.get_style_context().add_class('search-view-counter')
+
+        if count == 1:
+            counter_text = _('1 result')
+        else:
+            counter_text = _('%s results') % count
+        label = Gtk.Label(label=counter_text)
+        self.add(label)
+        self.show_all()
+
+
+class ResultRow(Gtk.ListBoxRow):
+    def __init__(self, msg, account, jid):
+        Gtk.ListBoxRow.__init__(self)
+        self.account = account
+        self.jid = jid
+        self.time = msg.time
+        self._client = app.get_client(account)
+
+        self.contact = self._client.get_module('Contacts').get_contact(jid)
+
+        self.log_line_id = msg.log_line_id
+        self.timestamp = msg.time
+        self.kind = msg.kind
+
+        self.type = 'contact'
+        if msg.kind == KindConstant.GC_MSG:
+            self.type = 'groupchat'
+
+        self._ui = get_builder('search_view.ui')
+        self.add(self._ui.result_row_grid)
+
+        kind = 'status'
+        contact_name = msg.contact_name
+        if msg.kind in (
+                KindConstant.SINGLE_MSG_RECV, KindConstant.CHAT_MSG_RECV):
+            kind = 'incoming'
+            contact_name = self.contact.name
+        elif msg.kind == KindConstant.GC_MSG:
+            kind = 'incoming'
+        elif msg.kind in (
+                KindConstant.SINGLE_MSG_SENT, KindConstant.CHAT_MSG_SENT):
+            kind = 'outgoing'
+            contact_name = app.nicks[account]
+        self._ui.row_name_label.set_text(contact_name)
+
+        avatar = self._get_avatar(kind, contact_name)
+        self._ui.row_avatar.set_from_surface(avatar)
+
+        local_time = time.localtime(msg.time)
+        date = time.strftime('%H:%M', local_time)
+        self._ui.row_time_label.set_label(date)
+
+        message_widget = MessageWidget(account)
+        self._ui.result_row_grid.attach(message_widget, 1, 1, 2, 1)
+        result = process(msg.message)
+        message_widget.add_content(result.blocks)
+
+        self.show_all()
+
+    def _get_avatar(self, kind, name):
+        scale = self.get_scale_factor()
+        if self.contact.is_groupchat:
+            contact = self.contact.get_resource(name)
+            return contact.get_avatar(AvatarSize.ROSTER, scale, add_show=False)
+
+        if kind == 'outgoing':
+            contact = self._client.get_module('Contacts').get_contact(
+                str(self._client.get_own_jid().bare))
+        else:
+            contact = self.contact
+
+        return contact.get_avatar(AvatarSize.ROSTER, scale, add_show=False)
diff --git a/gajim/gui_menu_builder.py b/gajim/gui_menu_builder.py
index 69047a85627cef774f03b1b11221b90636783821..9c75e0045168a65dfdbad8074fa978b9bd487f9e 100644
--- a/gajim/gui_menu_builder.py
+++ b/gajim/gui_menu_builder.py
@@ -525,6 +525,7 @@ def get_singlechat_menu(control_id, account, jid, type_):
         ('win.start-call-', _('Start Call…')),
         ('win.information-', _('Information')),
         ('app.browse-history', _('History')),
+        ('win.search-history', _('Search…')),
     ]
 
     def build_chatstate_menu():
@@ -549,7 +550,11 @@ def build_menu(preset):
                 if action_name == 'win.send-marker-' and type_ == 'pm':
                     continue
 
-                if action_name == 'app.browse-history':
+                if action_name == 'win.search-history':
+                    menuitem = Gio.MenuItem.new(label, action_name)
+                    menuitem.set_action_and_target_value(action_name, None)
+                    menu.append_item(menuitem)
+                elif action_name == 'app.browse-history':
                     menuitem = Gio.MenuItem.new(label, action_name)
                     dict_ = {'account': GLib.Variant('s', account),
                              'jid': GLib.Variant('s', str(jid))}
@@ -581,6 +586,7 @@ def get_groupchat_menu(control_id, account, jid):
         ('win.request-voice-', _('Request Voice')),
         ('win.execute-command-', _('Execute Command…')),
         ('app.browse-history', _('History')),
+        ('win.search-history', _('Search…')),
     ]
 
     def build_menu(preset):
@@ -588,7 +594,12 @@ def build_menu(preset):
         for item in preset:
             if isinstance(item[1], str):
                 action_name, label = item
-                if action_name == 'app.browse-history':
+                if action_name == 'win.search-history':
+                    menuitem = Gio.MenuItem.new(label, action_name)
+                    menuitem.set_action_and_target_value(action_name, None)
+                    menu.append_item(menuitem)
+
+                elif action_name == 'app.browse-history':
                     menuitem = Gio.MenuItem.new(label, action_name)
                     dict_ = {'account': GLib.Variant('s', account),
                              'jid': GLib.Variant('s', jid)}