diff --git a/README.md b/README.md
index d7998595b03edb03ad4559cf179a5632d7693f02..81e8e036e53bf71029e7ca6abe725cf993e2069f 100644
--- a/README.md
+++ b/README.md
@@ -24,7 +24,8 @@ NODE_ENV=development
     "VUE_APP_LOCALE":"fr-FR",
     "VUE_APP_APPLICATION_NAME":"GéoContrib",
     "VUE_APP_APPLICATION_ABSTRACT":"Application de saisie d'informations géographiques contributive",
-    "VUE_APP_LOGO_PATH":"@/assets/img/logo-neogeo-circle.png",
+    "VUE_APP_APPLICATION_FAVICO":"/geocontrib/favicon.ico",
+    "VUE_APP_LOGO_PATH":"/geocontrib/img/logo-neogeo-circle.png",
     "VUE_APP_DJANGO_BASE":"",
     "VUE_APP_DJANGO_API_BASE":"/geocontrib/api/",
     "DEFAULT_BASE_MAP":{
diff --git a/package-lock.json b/package-lock.json
index aac4365c145df2276e626c9152f58a9009190219..5c65244599ec38ec18636b5cfef2ee74c1468351 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -11592,6 +11592,11 @@
         }
       }
     },
+    "vue-multiselect": {
+      "version": "2.1.6",
+      "resolved": "https://registry.npmjs.org/vue-multiselect/-/vue-multiselect-2.1.6.tgz",
+      "integrity": "sha512-s7jmZPlm9FeueJg1RwJtnE9KNPtME/7C8uRWSfp9/yEN4M8XcS/d+bddoyVwVnvFyRh9msFo0HWeW0vTL8Qv+w=="
+    },
     "vue-router": {
       "version": "3.5.2",
       "resolved": "https://registry.npmjs.org/vue-router/-/vue-router-3.5.2.tgz",
diff --git a/package.json b/package.json
index 6288ae767b382e24b599d969739e7d3d9c545a0e..a508a7db81970ff804f9708136a3cd6252e274c4 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
 {
   "name": "geocontrib-frontend",
-  "version": "2.2.0",
+  "version": "2.3.0-rc1",
   "private": true,
   "scripts": {
     "serve": "npm run init-proxy & npm run init-serve",
@@ -27,6 +27,7 @@
     "sortablejs": "^1.14.0",
     "vue": "^2.6.11",
     "vue-frag": "^1.1.5",
+    "vue-multiselect": "~2.1.6",
     "vue-router": "^3.2.0",
     "vuex": "^3.6.2"
   },
diff --git a/public/config/config.json b/public/config/config.json
index 65038a32551262edcccf998e4c66e7707f44de04..f8f8e5834a5e41535a50f00884e924978e71979d 100644
--- a/public/config/config.json
+++ b/public/config/config.json
@@ -4,6 +4,7 @@
     "NODE_ENV":"development",
     "VUE_APP_LOCALE":"fr-FR",
     "VUE_APP_APPLICATION_NAME":"GéoContrib",
+    "VUE_APP_APPLICATION_FAVICO":"/geocontrib/img/geo2f.ico",
     "VUE_APP_APPLICATION_ABSTRACT":"Application de saisie d'informations géographiques contributive",
     "VUE_APP_LOGO_PATH":"/geocontrib/img/logo-neogeo-circle.png",
     "VUE_APP_DJANGO_BASE":"http://localhost:8010",
diff --git a/public/img/geo2f.ico b/public/img/geo2f.ico
new file mode 100644
index 0000000000000000000000000000000000000000..434ba30085b6358a339f1b89d4f16c57d2766750
Binary files /dev/null and b/public/img/geo2f.ico differ
diff --git a/public/img/logo_g2f.png b/public/img/logo_g2f.png
new file mode 100644
index 0000000000000000000000000000000000000000..4eedc26700eeaecf751908a8b0c2c4a22f79d96e
Binary files /dev/null and b/public/img/logo_g2f.png differ
diff --git a/src/assets/js/map-util.js b/src/assets/js/map-util.js
index 3c00b6df99f07e8c38df0e25701ec5539acc2781..84dc6f60b5cca045e97d1a2e65f1a02128c0d5d4 100644
--- a/src/assets/js/map-util.js
+++ b/src/assets/js/map-util.js
@@ -4,14 +4,9 @@
 import L from "leaflet"
 import "leaflet/dist/leaflet.css";
 import flip from '@turf/flip'
-import axios from "axios"
+import axios from '@/axios-client.js';
 import "leaflet.vectorgrid";
-
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value != null) ? unescape(value[1]) : null;
-})('csrftoken');
+import store from '@/store';
 
 import { FillSymbolizer, PointSymbolizer, LineSymbolizer } from "@/assets/js/vector_tile_fix.js";
 
@@ -389,12 +384,25 @@ const mapUtil = {
         // Look for a custom field
         let customField;
         let customFieldOption;
-        if (Object.keys(feature.properties).some(el => featureType.customfield_set && featureType.customfield_set.map(e => e.name).includes(el))) {
+        if (featureType.customfield_set && Object.keys(feature.properties).some(el => featureType.customfield_set.map(e => e.name).includes(el))) {
           customField = Object.keys(feature.properties).filter(el => featureType.customfield_set.map(e => e.name).includes(el));
           customFieldOption = feature.properties[customField[0]];
         }
-        const color = this.retrieveFeatureColor(featureType, feature.properties) || feature.properties.color;
-
+        let color = this.retrieveFeatureColor(featureType, feature.properties) || feature.properties.color;
+
+        // if (!color && customFieldOption && featureType.colors_style) {
+        //   color =
+        //   featureType.colors_style.value ?
+        //     featureType.colors_style.value.colors[customFieldOption].value ?
+        //     featureType.colors_style.value.colors[customFieldOption].value :
+        //     featureType.colors_style.value.colors[customFieldOption] :
+        //     featureType.colors_style.colors[customFieldOption]
+        // } else {
+        //   color = feature.properties.color;
+        // }
+        if (color == undefined){
+          color = featureType.color;
+        }
         if (geomJSON.type === 'Point') {
           if (customFieldOption && featureType.colors_style && featureType.colors_style.value && featureType.colors_style.value.icons) {
             const iconHTML = `
@@ -487,13 +495,13 @@ const mapUtil = {
 
     return `
 					<h4>
-						<a href="${feature_url}">${feature.properties.title}</a>
+						<a href="${store.state.configuration.BASE_URL.slice(0, -1)}${feature_url}">${feature.properties.title}</a>
 					</h4>
 					<div>
 						Statut : ${status}
 					</div>
 					<div>
-						Type : <a href="${feature_type_url}"> ${feature_type.title} </a>
+						Type : <a href="${store.state.configuration.BASE_URL.slice(0, -1)}${feature_type_url}"> ${feature_type.title} </a>
 					</div>
 					<div>
 						Dernière mise à jour : ${date_maj}
diff --git a/src/assets/styles/base.css b/src/assets/styles/base.css
index d49a7d4750077875a1d26d66f025feef2eba7b9d..0777e8e3742e08d03179550ddfff9e40778efb21 100644
--- a/src/assets/styles/base.css
+++ b/src/assets/styles/base.css
@@ -158,4 +158,48 @@ footer .ui.text.menu .item:not(:first-child) {
   border-radius: 3px;
   background-color: rgb(250, 241, 242);
   padding: 1rem;
+}
+
+/* ---------------------------------- */
+      /* ERROR LIST */
+/* ---------------------------------- */
+.multiselect__tags {
+  border: 2px solid #ced4da;
+}
+.multiselect__tags > .multiselect__input {
+  border: none !important;
+  font-size: 1rem !important;
+}
+
+.multiselect__placeholder {
+  color: #838383;
+  margin-bottom: 0px;
+  padding-top: 0;
+}
+
+.multiselect__single, .multiselect__tags, .multiselect__content, .multiselect__option {
+  text-overflow: ellipsis;
+  white-space: nowrap;
+  overflow: hidden;
+  max-width: 100%;
+}
+
+.multiselect__select {
+  z-index: 1 !important;
+}
+.multiselect__clear {
+  position: absolute;
+  right: 1px;
+  top: 8px;
+  width: 40px;
+  display: block;
+  cursor: pointer;
+  z-index: 9;
+  background-color: #fff;
+}
+.multiselect__spinner {
+  z-index: 2 !important;
+  background-color: #fff;
+  opacity: 1;
+  top: 2px;
 }
\ No newline at end of file
diff --git a/src/axios-client.js b/src/axios-client.js
new file mode 100644
index 0000000000000000000000000000000000000000..fa6b3d958ba2c8e4ec140a407d117a8137aee1c2
--- /dev/null
+++ b/src/axios-client.js
@@ -0,0 +1,35 @@
+import axios from 'axios';
+
+axios.defaults.withCredentials = true;
+
+// Add a request interceptor
+axios.interceptors.request.use(function (config) {
+
+  config.headers['X-CSRFToken'] = (name => {
+    const re = new RegExp(name + "=([^;]+)");
+    const value = re.exec(document.cookie);
+    return (value != null) ? unescape(value[1]) : null;
+  })('csrftoken');
+
+  return config;
+
+  }, function (error) {
+    return Promise.reject(error);
+  });
+
+// Add a response interceptor
+axios.interceptors.response.use(function (response) {
+
+  response.headers['X-CSRFToken'] = (name => {
+    const re = new RegExp(name + "=([^;]+)");
+    const value = re.exec(document.cookie);
+    return (value != null) ? unescape(value[1]) : null;
+  })('csrftoken');
+
+  return response;
+
+  }, function (error) {
+    return Promise.reject(error);
+  });
+
+export default axios;
\ No newline at end of file
diff --git a/src/components/SearchFeature.vue b/src/components/SearchFeature.vue
new file mode 100644
index 0000000000000000000000000000000000000000..6ea7c0d5747542584fedcddacede3988782670cc
--- /dev/null
+++ b/src/components/SearchFeature.vue
@@ -0,0 +1,114 @@
+<template>
+  <div>
+    <Multiselect
+      v-model="selection"
+      :options="results"
+      :options-limit="10"
+      :allow-empty="true"
+      track-by="feature_id"
+      label="title"
+      :reset-after="false"
+      select-label=""
+      selected-label=""
+      deselect-label=""
+      :searchable="true"
+      :placeholder="'Rechercher un signalement ...'"
+      :show-no-results="true"
+      :loading="loading"
+      :clear-on-select="false"
+      :preserve-search="true"
+      @search-change="search"
+      @select="select"
+      @close="close"
+    >
+      <template slot="clear">
+        <div
+          v-if="selection"
+          class="multiselect__clear"
+          @click.prevent.stop="selection = null"
+        >
+          <i class="close icon"></i>
+        </div>
+      </template>
+      <span slot="noResult">
+        Aucun résultat.
+      </span>
+      <span slot="noOptions">
+        Saisissez les premiers caractères ...
+      </span>
+    </Multiselect>
+  </div>
+</template>
+
+<script>
+import { mapState, mapMutations, mapActions } from 'vuex';
+
+import Multiselect from 'vue-multiselect';
+
+export default {
+  name: 'SearchFeature',
+
+  components: {
+    Multiselect
+  },
+
+  data() {
+    return {
+      loading: false,
+      selection: null,
+      text: null,
+      results: []
+    }
+  },
+
+  computed: {
+    ...mapState('feature', [
+      'features'
+    ])
+  },
+
+  watch: {
+    text: function(newValue) {
+      this.loading = true;
+      this.GET_PROJECT_FEATURES({
+        project_slug: this.$route.params.slug,
+        feature_type__slug: this.$route.params.slug_type_signal,
+        search: newValue,
+        limit: '10'
+      })
+        .then(() => {
+          if (newValue) {
+            this.results = this.features;
+          } else {
+            this.results.splice(0);
+          }
+          this.loading = false;
+        });
+    }
+  },
+
+  created() {
+    this.RESET_CANCELLABLE_SEARCH_REQUEST();
+  },
+
+  methods: {
+    ...mapMutations(['RESET_CANCELLABLE_SEARCH_REQUEST']),
+    ...mapActions('feature', [
+      'GET_PROJECT_FEATURES'
+    ]),
+    search(text) {
+      this.text = text;
+    },
+    select(e) {
+      this.$emit('select', e);
+    },
+    close() {
+      this.$emit('close', this.selection);
+    }
+  }
+}
+</script>
+
+<style scoped>
+
+</style>
diff --git a/src/components/feature/FeatureAttachmentForm.vue b/src/components/feature/FeatureAttachmentForm.vue
index beb3e6657bb4630b12fa3fc394be9b22528898df..26d93d3ac930559f7ed63bcad41f1ec92a3a5d9e 100644
--- a/src/components/feature/FeatureAttachmentForm.vue
+++ b/src/components/feature/FeatureAttachmentForm.vue
@@ -173,12 +173,13 @@ export default {
       let image = new Image();
       image.onload = function () {
         handleFile(true);
+        URL.revokeObjectURL(image.src);
       };
       image.onerror = function () {
         handleFile(false);
+        URL.revokeObjectURL(image.src);
       };
       image.src = url.createObjectURL(files);
-      URL.revokeObjectURL(image.src);
     },
 
     onFileChange(e) {
diff --git a/src/components/feature/FeatureLinkedForm.vue b/src/components/feature/FeatureLinkedForm.vue
index 02cd0bfe25a382e920681ea126841bc5ac80f971..27c60fbd5eaa879ae80d94c40df07bb04de5cc56 100644
--- a/src/components/feature/FeatureLinkedForm.vue
+++ b/src/components/feature/FeatureLinkedForm.vue
@@ -30,10 +30,9 @@
           <label for="form.feature_to.id_for_label">{{
             form.feature_to.label
           }}</label>
-          <Dropdown
-            :options="featureOptions"
-            :selected="selected_feature_to"
-            :selection.sync="selected_feature_to"
+          <SearchFeature
+            @select="selectFeatureTo"
+            @close="selectFeatureTo"
           />
           {{ form.feature_to.errors }}
         </div>
@@ -44,6 +43,7 @@
 
 <script>
 import Dropdown from "@/components/Dropdown.vue";
+import SearchFeature from '@/components/SearchFeature.vue';
 
 export default {
   name: "FeatureLinkedForm",
@@ -52,6 +52,7 @@ export default {
 
   components: {
     Dropdown,
+    SearchFeature
   },
 
   data() {
@@ -92,22 +93,6 @@ export default {
   },
 
   computed: {
-    featureOptions: function () {
-      return this.features
-        .filter(
-          (el) =>
-            el.feature_type.slug === this.$route.params.slug_type_signal && //* filter only for the same feature
-            el.feature_id !== this.$route.params.slug_signal //* filter out current feature
-        )
-        .map((el) => {
-          return {
-            name: `${el.title} (${el.display_creator} - ${this.formatDate(
-              el.created_on
-            )})`,
-            value: el.feature_id,
-          };
-        });
-    },
 
     selected_relation_type: {
       // getter
@@ -119,27 +104,7 @@ export default {
         this.form.relation_type.value = newValue;
         this.updateStore();
       },
-    },
-
-    selected_feature_to: {
-      // getter
-      get() {
-        return this.form.feature_to.value.name;
-      },
-      // setter
-      set(newValue) {
-        this.form.feature_to.value = newValue;
-        this.updateStore();
-      },
-    },
-  },
-
-  watch: {
-    featureOptions(newValue) {
-      if (newValue) {
-        this.getExistingFeature_to(newValue);
-      }
-    },
+    }
   },
 
   methods: {
@@ -153,6 +118,10 @@ export default {
       this.$store.commit("feature/REMOVE_LINKED_FORM", this.linkedForm.dataKey);
     },
 
+    selectFeatureTo(e) {
+      this.form.feature_to.value = e;
+    },
+
     updateStore() {
       this.$store.commit("feature/UPDATE_LINKED_FORM", {
         dataKey: this.linkedForm.dataKey,
diff --git a/src/components/map-layers/SidebarLayers.vue b/src/components/map-layers/SidebarLayers.vue
index dace42eff5f205a37bc614755fcb71dde1d1253d..950f41b95854e32a2bdaae7099eaacbb6d5701c4 100644
--- a/src/components/map-layers/SidebarLayers.vue
+++ b/src/components/map-layers/SidebarLayers.vue
@@ -220,13 +220,20 @@ export default {
 
     initSortable() {
       this.baseMaps.forEach((basemap) => {
-        new Sortable(document.getElementById(`list-${basemap.id}`), {
-          animation: 150,
-          handle: ".layer-handle-sort", // The element that is active to drag
-          ghostClass: "blue-background-class",
-          dragClass: "white-opacity-background-class",
-          onEnd: this.onlayerMove.bind(this),
-        });
+        let element=document.getElementById(`list-${basemap.id}`);
+        if(element) {
+          new Sortable(element, {
+            animation: 150,
+            handle: ".layer-handle-sort", // The element that is active to drag
+            ghostClass: "blue-background-class",
+            dragClass: "white-opacity-background-class",
+            onEnd: this.onlayerMove.bind(this),
+          });
+        }
+        else{
+          console.error(`list-${basemap.id} not found in dom`)
+        }
+
       });
     },
     // Check if there are changes in the basemaps settings. Changes are detected if:
diff --git a/src/main.js b/src/main.js
index 3e5f6956593ac0d53834bdcd46cf80970188613a..56bc1b0585e4d9d65e0712b9b6438479597db717 100644
--- a/src/main.js
+++ b/src/main.js
@@ -13,6 +13,8 @@ import '@fortawesome/fontawesome-free/js/all.js'
 import { library } from '@fortawesome/fontawesome-svg-core';
 import { fas } from '@fortawesome/free-solid-svg-icons';
 import { FontAwesomeIcon } from '@fortawesome/vue-fontawesome';
+// Multiselect installation
+import 'vue-multiselect/dist/vue-multiselect.min.css';
 
 library.add(fas)
 
@@ -36,6 +38,15 @@ if(navigator.serviceWorker){
 
 let onConfigLoaded = function(config){
   store.commit("SET_CONFIG", config);
+
+  // set title and favico
+  document.title= config.VUE_APP_APPLICATION_NAME+' '+config.VUE_APP_APPLICATION_ABSTRACT;
+  let link = document.createElement('link');
+  link.id = 'dynamic-favicon';
+  link.rel = 'shortcut icon';
+  link.href = config.VUE_APP_APPLICATION_FAVICO;
+  document.head.appendChild(link);
+
   window.proxy_url=config.VUE_APP_DJANGO_API_BASE+"proxy/";
   axios.all([store.dispatch("USER_INFO"),
     store.dispatch("GET_ALL_PROJECTS"),
diff --git a/src/service-worker.js b/src/service-worker.js
index c9c0d32fee26672dbdc0b2f24959aee7e9eb3381..dac6068b2cb19d00474c971971347d4f6fbb87a1 100644
--- a/src/service-worker.js
+++ b/src/service-worker.js
@@ -14,7 +14,7 @@ if (workbox) {
     // Since we have a SPA here, this should be index.html always.
     // https://stackoverflow.com/questions/49963982/vue-router-history-mode-with-pwa-in-offline-mode
     workbox.routing.registerNavigationRoute('/geocontrib/index.html', {
-        blacklist: [/\/api/,/\/admin/],
+        blacklist: [/\/api/,/\/admin/,/\/media/],
       })
 
     workbox.routing.registerRoute(
diff --git a/src/services/map-api.js b/src/services/map-api.js
index fa4c5ddba7178edc5d26a91095d1dedb3c976e46..f4b8f45edf3b0f2d8835344f0fcb088f45059c88 100644
--- a/src/services/map-api.js
+++ b/src/services/map-api.js
@@ -1,11 +1,6 @@
-import axios from 'axios';
+import axios from '@/axios-client.js';
 import store from '../store';
 
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value !== null) ? unescape(value[1]) : null;
-})('csrftoken');
 
 const baseUrl = store.state.configuration.VUE_APP_DJANGO_API_BASE;
 
diff --git a/src/services/project-api.js b/src/services/project-api.js
index b918ad2270c8dbcd20cd0d46f8c903af3176a3a4..f3d439cd4b4648a8080ec68e540dfbcf451d8810 100644
--- a/src/services/project-api.js
+++ b/src/services/project-api.js
@@ -1,12 +1,7 @@
-import axios from 'axios';
+import axios from '@/axios-client.js';
 import store from '../store'
 
 
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value !== null) ? unescape(value[1]) : null;
-})('csrftoken');
 
 
 const baseUrl = store.state.configuration.VUE_APP_DJANGO_API_BASE;
diff --git a/src/store/index.js b/src/store/index.js
index eea20cf397445f8aa14c8dcaec92552347a08168..5e7e3c39ae6a7d0840790123e03d1b079c589068 100644
--- a/src/store/index.js
+++ b/src/store/index.js
@@ -1,5 +1,4 @@
-const axios = require("axios");
-
+import axios from '@/axios-client.js';
 import Vue from 'vue';
 import Vuex from 'vuex';
 import router from '../router'
@@ -7,26 +6,26 @@ import feature_type from "./modules/feature_type"
 import feature from "./modules/feature"
 import map from "./modules/map"
 
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value !== null) ? unescape(value[1]) : null;
-})('csrftoken');
+// axios.defaults.headers.common['X-CSRFToken'] = (name => {
+//   var re = new RegExp(name + "=([^;]+)");
+//   var value = re.exec(document.cookie);
+//   return (value !== null) ? unescape(value[1]) : null;
+// })('csrftoken');
 
 
 Vue.use(Vuex);
 
 
-axios.defaults.withCredentials = true; // * add cookies to axios
-function updateAxiosHeader() {
-  axios.defaults.headers.common['X-CSRFToken'] = (name => {
-    var re = new RegExp(name + "=([^;]+)");
-    var value = re.exec(document.cookie);
-    return (value !== null) ? unescape(value[1]) : null;
-  })('csrftoken');
-}
-// ! À vérifier s'il y a un changement de token pendant l'éxécution de l'appli
-updateAxiosHeader();
+// axios.defaults.withCredentials = true; // * add cookies to axios
+// function updateAxiosHeader() {
+//   axios.defaults.headers.common['X-CSRFToken'] = (name => {
+//     var re = new RegExp(name + "=([^;]+)");
+//     var value = re.exec(document.cookie);
+//     return (value !== null) ? unescape(value[1]) : null;
+//   })('csrftoken');
+// }
+// // ! À vérifier s'il y a un changement de token pendant l'éxécution de l'appli
+// updateAxiosHeader();
 
 const noPermissions = { "can_view_project": true, "can_create_project": false, "can_update_project": false, "can_view_feature": true, "can_view_archived_feature": true, "can_create_feature": false, "can_update_feature": false, "can_delete_feature": false, "can_publish_feature": false, "can_create_feature_type": false, "can_view_feature_type": true, "is_project_administrator": false }
 
@@ -53,6 +52,7 @@ export default new Vuex.Store({
       isLoading: false,
       message: "En cours de chargement"
     },
+    cancellableSearchRequest: []
   },
 
   mutations: {
@@ -117,6 +117,13 @@ export default new Vuex.Store({
         message: "En cours de chargement"
       };
     },
+    SET_CANCELLABLE_SEARCH_REQUEST(state, payload) {
+      state.cancellableSearchRequest.push(payload);
+    },
+  
+    RESET_CANCELLABLE_SEARCH_REQUEST(state) {
+      state.cancellableSearchRequest = [];
+    },
   },
 
   getters: {
@@ -274,16 +281,17 @@ export default new Vuex.Store({
         });
     },
 
-    async GET_PROJECT_INFO({ state, commit, dispatch }, slug, noFeatures) {
+    async GET_PROJECT_INFO({ state, commit, dispatch }, slug/*, noFeatures */) {
       commit("SET_PROJECT_SLUG", slug);
       let promises = [
         dispatch("GET_PROJECT_LAST_MESSAGES", slug).then(response => response),
         dispatch("feature_type/GET_PROJECT_FEATURE_TYPES", slug).then(response => response),
+        // dispatch("feature/GET_PROJECT_FEATURES", slug).then(response => response),
       ]
-      console.log(noFeatures)
+     /*  console.log(noFeatures)
       if (!noFeatures) {
         promises.push(dispatch("feature/GET_PROJECT_FEATURES", slug).then(response => response))
-      }
+      } */
       console.log(promises)
       if (state.user) promises.push(dispatch("map/GET_BASEMAPS", slug).then(response => response))
 
diff --git a/src/store/modules/feature.js b/src/store/modules/feature.js
index 2b9eb239c3eded5a247a1572731689d54e219a94..d081b0b6b49432a88d697d0434f0871b913466c2 100644
--- a/src/store/modules/feature.js
+++ b/src/store/modules/feature.js
@@ -1,11 +1,11 @@
-const axios = require("axios");
+import axios from '@/axios-client.js';
 import router from '../../router'
 
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value !== null) ? unescape(value[1]) : null;
-})('csrftoken');
+// axios.defaults.headers.common['X-CSRFToken'] = (name => {
+//   var re = new RegExp(name + "=([^;]+)");
+//   var value = re.exec(document.cookie);
+//   return (value !== null) ? unescape(value[1]) : null;
+// })('csrftoken');
 
 
 const feature = {
@@ -99,9 +99,28 @@ const feature = {
   getters: {
   },
   actions: {
-    GET_PROJECT_FEATURES({ commit, rootState }, project_slug) {
+    GET_PROJECT_FEATURES({ commit, rootState }, { project_slug, feature_type__slug, search, limit }) {
+      if (rootState.cancellableSearchRequest.length > 0) {
+        const currentRequestCancelToken =
+          rootState.cancellableSearchRequest[rootState.cancellableSearchRequest.length - 1];
+        currentRequestCancelToken.cancel();
+      }
+  
+      const cancelToken = axios.CancelToken.source();
+      commit('SET_CANCELLABLE_SEARCH_REQUEST', cancelToken, { root: true });
+      commit("SET_FEATURES", []);
+      let url = `${rootState.configuration.VUE_APP_DJANGO_API_BASE}projects/${project_slug}/feature/`;
+      if (feature_type__slug) {
+        url = url.concat('', `${url.includes('?') ? '&' : '?'}feature_type__slug=${feature_type__slug}`);
+      }
+      if (search) {
+        url = url.concat('', `${url.includes('?') ? '&' : '?'}title__contains=${search}`);
+      }
+      if (limit) {
+        url =url.concat('', `${url.includes('?') ? '&' : '?'}limit=${limit}`);
+      }
       return axios
-        .get(`${rootState.configuration.VUE_APP_DJANGO_API_BASE}projects/${project_slug}/feature/`)
+        .get(url , { cancelToken: cancelToken.token })
         .then((response) => {
           if (response.status === 200 && response.data) {
             const features = response.data.features;
@@ -120,18 +139,24 @@ const feature = {
       const message = routeName === "editer-signalement" ? "Le signalement a été mis à jour" : "Le signalement a été crée";
 
       function redirect(featureId) {
-        dispatch("GET_PROJECT_FEATURES", rootState.project_slug).then(() => {
-          console.log(state.feature);
-          commit("DISCARD_LOADER", null, { root: true })
-          router.push({
-            name: "details-signalement",
-            params: {
-              slug_type_signal: rootState.feature_type.current_feature_type_slug,
-              slug_signal: featureId,
-              message,
-            },
+        dispatch(
+          'GET_PROJECT_FEATURES',
+          {
+            project_slug: rootState.project_slug,
+            feature_type__slug: rootState.feature_type.current_feature_type_slug
+          }
+        )
+          .then(() => {
+            commit("DISCARD_LOADER", null, { root: true })
+            router.push({
+              name: "details-signalement",
+              params: {
+                slug_type_signal: rootState.feature_type.current_feature_type_slug,
+                slug_signal: featureId,
+                message,
+              },
+            });
           });
-        })
       }
 
       async function handleOtherForms(featureId) {
diff --git a/src/store/modules/feature_type.js b/src/store/modules/feature_type.js
index bed583d904949de35706a19a93ea4d10a95e725d..cfc3c0ddb6f6d5d20bd0133ac18b4c046ee99b9a 100644
--- a/src/store/modules/feature_type.js
+++ b/src/store/modules/feature_type.js
@@ -1,10 +1,10 @@
-import axios from "axios"
+import axios from '@/axios-client.js';
 
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value !== null) ? unescape(value[1]) : null;
-})('csrftoken');
+// axios.defaults.headers.common['X-CSRFToken'] = (name => {
+//   var re = new RegExp(name + "=([^;]+)");
+//   var value = re.exec(document.cookie);
+//   return (value !== null) ? unescape(value[1]) : null;
+// })('csrftoken');
 
 const getColorsStyles = (customForms) => customForms.filter(customForm => customForm.options && customForm.options.length).map(el => {
   //* in dropdown, value is the name and name is the label to be displayed, could be changed...
diff --git a/src/store/modules/map.js b/src/store/modules/map.js
index c8e5ce93eab03a485b06f5a409be781a95cbe313..b9843087d53556c34c8fd0b7831b5e4560755305 100644
--- a/src/store/modules/map.js
+++ b/src/store/modules/map.js
@@ -1,11 +1,11 @@
-const axios = require("axios");
+import axios from '@/axios-client.js';
 import { mapUtil } from "@/assets/js/map-util.js";
 
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value !== null) ? unescape(value[1]) : null;
-})('csrftoken');
+// axios.defaults.headers.common['X-CSRFToken'] = (name => {
+//   var re = new RegExp(name + "=([^;]+)");
+//   var value = re.exec(document.cookie);
+//   return (value !== null) ? unescape(value[1]) : null;
+// })('csrftoken');
 
 
 const map = {
diff --git a/src/views/feature/Feature_detail.vue b/src/views/feature/Feature_detail.vue
index 24514f13d3786d43b7de0a829c2011b413a2a135..416703ec722b0be57646869861ec0da713ad0e22 100644
--- a/src/views/feature/Feature_detail.vue
+++ b/src/views/feature/Feature_detail.vue
@@ -351,13 +351,13 @@ import frag from "vue-frag";
 import { mapGetters, mapState } from "vuex";
 import { mapUtil } from "@/assets/js/map-util.js";
 import featureAPI from "@/services/feature-api";
-const axios = require("axios");
+import axios from '@/axios-client.js';
 
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value !== null) ? unescape(value[1]) : null;
-})('csrftoken');
+// axios.defaults.headers.common['X-CSRFToken'] = (name => {
+//   var re = new RegExp(name + "=([^;]+)");
+//   var value = re.exec(document.cookie);
+//   return (value !== null) ? unescape(value[1]) : null;
+// })('csrftoken');
 
 export default {
   name: "Feature_detail",
@@ -445,7 +445,6 @@ export default {
       return status ? status.name : "";
     },
   },
-
   filters: {
     formatDate(value) {
       let date = new Date(value);
@@ -509,12 +508,13 @@ export default {
       let image = new Image();
       image.onload = function () {
         handleFile(true);
+        URL.revokeObjectURL(image.src);
       };
       image.onerror = function () {
         handleFile(false);
+        URL.revokeObjectURL(image.src);
       };
       image.src = url.createObjectURL(files);
-      URL.revokeObjectURL(image.src);
     },
 
     onFileChange(e) {
@@ -567,7 +567,9 @@ export default {
           if (response.status === 204) {
             this.$store.dispatch(
               "feature/GET_PROJECT_FEATURES",
-              this.$route.params.slug
+              {
+                project_slug: this.$route.params.slug
+              }
             );
             this.goBackToProject();
           }
@@ -683,8 +685,13 @@ export default {
   mounted() {
     this.$store.commit("DISPLAY_LOADER", "Recherche du signalement");
     if (!this.project) {
-      this.$store
-        .dispatch("GET_PROJECT_INFO", this.$route.params.slug)
+      // Chargements des features et infos projet en cas d'arrivée directe sur la page ou de refresh
+      axios.all([
+        this.$store
+        .dispatch("GET_PROJECT_INFO", this.$route.params.slug),
+        this.$store.dispatch('feature/GET_PROJECT_FEATURES', {
+          project_slug: this.$route.params.slug
+        })])
         .then(() => {
           this.$store.commit("DISCARD_LOADER");
           this.initMap();
diff --git a/src/views/feature/Feature_edit.vue b/src/views/feature/Feature_edit.vue
index 529584b5452c7d94e04a0860afeda65010aacce8..30ceab1a5636a89b695fe2f8c4991fa3b40ec77e 100644
--- a/src/views/feature/Feature_edit.vue
+++ b/src/views/feature/Feature_edit.vue
@@ -274,14 +274,14 @@ import featureAPI from "@/services/feature-api";
 import L from "leaflet";
 import "leaflet-draw";
 import { mapUtil } from "@/assets/js/map-util.js";
-const axios = require("axios");
+import axios from '@/axios-client.js';
 import flip from "@turf/flip";
 
-axios.defaults.headers.common["X-CSRFToken"] = ((name) => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return value !== null ? unescape(value[1]) : null;
-})("csrftoken");
+// axios.defaults.headers.common["X-CSRFToken"] = ((name) => {
+//   var re = new RegExp(name + "=([^;]+)");
+//   var value = re.exec(document.cookie);
+//   return value !== null ? unescape(value[1]) : null;
+// })("csrftoken");
 
 export default {
   name: "Feature_edit",
@@ -1018,17 +1018,19 @@ export default {
   mounted() {
     this.$store
       .dispatch("GET_PROJECT_INFO", this.$route.params.slug)
-      .then((data) => {
-        console.log(data);
-        this.initForm();
-        this.initMap();
-        this.onFeatureTypeLoaded();
-        this.initExtraForms(this.feature);
-
-        setTimeout(() => {
-          mapUtil.addGeocoders(this.$store.state.configuration);
-        }, 1000);
-      });
+        .then(() => {
+          this.initForm();
+          this.initMap();
+          this.onFeatureTypeLoaded();
+          this.initExtraForms(this.feature);
+
+          setTimeout(
+            function () {
+              mapUtil.addGeocoders(this.$store.state.configuration);
+            }.bind(this),
+            1000
+          );
+        });
   },
 
   destroyed() {
diff --git a/src/views/feature/Feature_list.vue b/src/views/feature/Feature_list.vue
index 51435ead9d232e1dbd97355efe483d8b64c9ee6b..f62de2ccefdc53b6cc17c2029b50cde0af788de7 100644
--- a/src/views/feature/Feature_list.vue
+++ b/src/views/feature/Feature_list.vue
@@ -191,13 +191,13 @@ import { mapUtil } from "@/assets/js/map-util.js";
 import SidebarLayers from "@/components/map-layers/SidebarLayers";
 import FeatureListTable from "@/components/feature/FeatureListTable";
 import Dropdown from "@/components/Dropdown.vue";
-const axios = require("axios");
+import axios from '@/axios-client.js';
 
-axios.defaults.headers.common["X-CSRFToken"] = ((name) => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return value !== null ? unescape(value[1]) : null;
-})("csrftoken");
+// axios.defaults.headers.common['X-CSRFToken'] = (name => {
+//   var re = new RegExp(name + "=([^;]+)");
+//   var value = re.exec(document.cookie);
+//   return (value !== null) ? unescape(value[1]) : null;
+// })('csrftoken');
 
 export default {
   name: "Feature_list",
@@ -318,7 +318,11 @@ export default {
         .then(() => {
           if (!this.modalAllDeleteOpen) {
             this.$store
-              .dispatch("feature/GET_PROJECT_FEATURES", this.project.slug) // ? Toujours nécessaire de mettre à jour les features classiques ?
+              .dispatch("feature/GET_PROJECT_FEATURES",
+                {
+                  project_slug: this.project.slug
+                }
+              )
               .then(() => {
                 this.updateFeatures();
                 this.checkedFeatures.splice(feature_id);
@@ -366,11 +370,13 @@ export default {
       this.zoom = this.$route.query.zoom || "";
       this.lat = this.$route.query.lat || "";
       this.lng = this.$route.query.lng || "";
+      
       var mapDefaultViewCenter =
         this.$store.state.configuration.DEFAULT_MAP_VIEW.center;
       var mapDefaultViewZoom =
         this.$store.state.configuration.DEFAULT_MAP_VIEW.zoom;
 
+      
       this.map = mapUtil.createMap(this.$refs.map, {
         zoom: this.zoom,
         lat: this.lat,
@@ -591,7 +597,7 @@ export default {
     },
   },
 
-  created() {
+/*   created() {
     if (!this.project) {
       this.$store.dispatch(
         "GET_PROJECT_INFO",
@@ -599,10 +605,24 @@ export default {
         "noFeatures" //* not fetching classic features, too heavy, using paginated features instead
       );
     }
-  },
+  }, */
 
   mounted() {
-    this.initMap();
+    if (!this.project) {
+      // Chargements des features et infos projet en cas d'arrivée directe sur la page ou de refresh
+      axios.all([
+        this.$store.dispatch("GET_PROJECT_INFO", this.$route.params.slug),
+        this.$store.dispatch('feature/GET_PROJECT_FEATURES', {
+          project_slug: this.$route.params.slug
+        })])
+        .then(() => {
+          this.initMap();
+        });
+    }
+    else {
+      this.initMap();
+    }
+    
   },
 
   destroyed() {
diff --git a/src/views/project/Project_detail.vue b/src/views/project/Project_detail.vue
index 56b7bd35dd3439318ba4ea878f02b41054b9c156..2e7066dd2d7090c0176df277e063a6f34fb0cc60 100644
--- a/src/views/project/Project_detail.vue
+++ b/src/views/project/Project_detail.vue
@@ -337,6 +337,14 @@
               <div class="content">
                 <div class="center aligned header">Derniers signalements</div>
                 <div class="center aligned description">
+                  <div
+                    :class="{ active: featuresLoading }"
+                    class="ui inverted dimmer"
+                  >
+                    <div class="ui text loader">
+                      Récupération des signalements en cours...
+                    </div>
+                  </div>
                   <div class="ui relaxed list">
                     <div
                       v-for="(item, index) in last_features"
@@ -387,7 +395,7 @@
                     >
                       <div class="content">
                         <div>
-                          <router-link :to="item.related_feature.feature_url"
+                          <router-link :to="getRouteUrl(item.related_feature.feature_url)"
                             >"{{ item.comment }}"</router-link
                           >
                         </div>
@@ -532,13 +540,13 @@ import { mapUtil } from "@/assets/js/map-util.js";
 import { mapGetters, mapState } from "vuex";
 import projectAPI from "@/services/project-api";
 
-const axios = require("axios");
+import axios from '@/axios-client.js';
 
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value !== null) ? unescape(value[1]) : null;
-})('csrftoken');
+// axios.defaults.headers.common['X-CSRFToken'] = (name => {
+//   var re = new RegExp(name + "=([^;]+)");
+//   var value = re.exec(document.cookie);
+//   return (value !== null) ? unescape(value[1]) : null;
+// })('csrftoken');
 
 export default {
   name: "Project_details",
@@ -572,6 +580,7 @@ export default {
       is_suscriber: false,
       tempMessage: null,
       featureTypeLoading: true,
+      featuresLoading: true
     };
   },
 
@@ -596,7 +605,9 @@ export default {
     refreshId() {
       return "?ver=" + Math.random();
     },
-
+    getRouteUrl(url){
+      return '/'+url.replace(this.$store.state.configuration.BASE_URL,''); // remove duplicate /geocontrib
+    },
     isOffline() {
       return navigator.onLine === false;
     },
@@ -771,10 +782,17 @@ export default {
   },
 
   mounted() {
-    this.$store.dispatch("GET_PROJECT_INFO", this.slug).then(() => {
-      this.featureTypeLoading = false;
-      setTimeout(this.initMap, 1000);
-    });
+    this.$store.dispatch('GET_PROJECT_INFO', this.slug)
+      .then(() => {
+        this.featureTypeLoading = false;
+        setTimeout(this.initMap, 1000);
+      });
+    this.$store.dispatch('feature/GET_PROJECT_FEATURES', {
+      project_slug: this.slug
+    })
+      .then(() => {
+        this.featuresLoading = false;
+      });
 
     if (this.message) {
       this.tempMessage = this.message;
diff --git a/src/views/project/Project_edit.vue b/src/views/project/Project_edit.vue
index 4779af2c603dd93c5694cb7646a466b049bdd265..f9784c6a4e5a064f2201ae0b7cc50c30a14cb556 100644
--- a/src/views/project/Project_edit.vue
+++ b/src/views/project/Project_edit.vue
@@ -191,16 +191,16 @@
 </template>
 
 <script>
-const axios = require("axios");
+import axios from '@/axios-client.js';
 import Dropdown from "@/components/Dropdown.vue";
 
 import { mapGetters } from "vuex";
 
-axios.defaults.headers.common["X-CSRFToken"] = ((name) => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return value !== null ? unescape(value[1]) : null;
-})("csrftoken");
+// axios.defaults.headers.common["X-CSRFToken"] = ((name) => {
+//   var re = new RegExp(name + "=([^;]+)");
+//   var value = re.exec(document.cookie);
+//   return value !== null ? unescape(value[1]) : null;
+// })("csrftoken");
 
 export default {
   name: "Project_edit",
@@ -287,12 +287,13 @@ export default {
       let image = new Image();
       image.onload = function () {
         handleFile(true);
+        URL.revokeObjectURL(image.src);
       };
       image.onerror = function () {
         handleFile(false);
+        URL.revokeObjectURL(image.src);
       };
       image.src = url.createObjectURL(files);
-      URL.revokeObjectURL(image.src);
     },
 
     onFileChange(e) {
diff --git a/src/views/project/Project_members.vue b/src/views/project/Project_members.vue
index 34c89ac5718fa7b0dd64f1eeb0e3ac2ad83865d9..aea8f4b1340c1ea3ed81f8ee426985195632eb74 100644
--- a/src/views/project/Project_members.vue
+++ b/src/views/project/Project_members.vue
@@ -106,17 +106,11 @@
 </template>
 
 <script>
-import axios from "axios";
+import axios from '@/axios-client.js';
 import frag from "vue-frag";
 import { mapGetters } from "vuex";
 import Dropdown from "@/components/Dropdown.vue";
 
-axios.defaults.headers.common['X-CSRFToken'] = (name => {
-  var re = new RegExp(name + "=([^;]+)");
-  var value = re.exec(document.cookie);
-  return (value !== null) ? unescape(value[1]) : null;
-})('csrftoken');
-
 export default {
   name: "Project_members",
 
@@ -296,7 +290,7 @@ export default {
       // todo: move function to a service
       return axios
         .get(
-          `${this.$store.state.configuration.VUE_APP_DJANGO_API_BASE}projects/${this.$route.params.slug}/utilisateurs`
+          `${this.$store.state.configuration.VUE_APP_DJANGO_API_BASE}projects/${this.$route.params.slug}/utilisateurs/`
         )
         .then((response) => response.data)
         .catch((error) => {
diff --git a/src/views/registration/Login.vue b/src/views/registration/Login.vue
index 79fe8a823ddb8ff58e2b70e1bb66a06d06ea80bb..0f62078398f1fe07998e86e91d73c4c2601cc8cf 100644
--- a/src/views/registration/Login.vue
+++ b/src/views/registration/Login.vue
@@ -4,7 +4,7 @@
       <div class="fourteen wide column">
         <img
           class="ui centered small image"
-          src="@/assets/img/logo-neogeo-circle.png"
+          :src="logo"
         />
         <h2 class="ui center aligned icon header">
           <div class="content">
@@ -74,13 +74,13 @@ export default {
     };
   },
   computed: {
-    LOGO_PATH: function () {
+    logo() {
       return this.$store.state.configuration.VUE_APP_LOGO_PATH;
     },
-    APPLICATION_NAME: function () {
+    APPLICATION_NAME() {
       return this.$store.state.configuration.VUE_APP_APPLICATION_NAME;
     },
-    APPLICATION_ABSTRACT: function () {
+    APPLICATION_ABSTRACT() {
       return this.$store.state.configuration.VUE_APP_APPLICATION_ABSTRACT;
     },
   },
diff --git a/vue.config.js b/vue.config.js
index 777fbbbaf0383828440b5391b142cb44add79982..f9d74727c971c84a1b1afb45270b6a2162a71b69 100644
--- a/vue.config.js
+++ b/vue.config.js
@@ -23,6 +23,14 @@ module.exports = {
                 /manifest\.json$/ 
             ],
         },
+        iconPaths: {
+            faviconSVG: null,
+            favicon32: null,
+            favicon16: null,
+            appleTouchIcon: null,
+            maskIcon: null,
+            msTileImage: null,
+          },
         themeColor: '#1da025'
       },
     configureWebpack: {