Lene Scholtz 6 gadus atpakaļ
vecāks
revīzija
6d061a034e
45 mainītis faili ar 2112 papildinājumiem un 925 dzēšanām
  1. Binārs
      public/img/icons/area.png
  2. Binārs
      public/img/icons/garage.png
  3. Binārs
      public/img/sort-down.png
  4. Binārs
      public/img/sort-up.png
  5. 3
    0
      src/App.vue
  6. 206
    1
      src/components/admin/misc/carousel.vue
  7. 1
    16
      src/components/admin/misc/carouselList.vue
  8. 101
    0
      src/components/admin/misc/carouselSearch.vue
  9. 5
    16
      src/components/admin/status/statusPage.vue
  10. 8
    4
      src/components/admin/status/tenderWeekAdminPage.vue
  11. 23
    11
      src/components/admin/status/timeshareAdminPage.vue
  12. 7
    3
      src/components/communication/templateDetail.vue
  13. 1
    0
      src/components/communication/templateInnerItem.vue
  14. 10
    6
      src/components/communication/templatePage.vue
  15. 17
    9
      src/components/home/carouselSection.vue
  16. 4
    0
      src/components/processFlow/makeOffer.vue
  17. 14
    4
      src/components/property/propertyCard.vue
  18. 107
    60
      src/components/property/propertyCreate.vue
  19. 12
    7
      src/components/property/propertyFieldEditor.vue
  20. 16
    4
      src/components/property/propertyImage.vue
  21. 33
    84
      src/components/property/propertyList.vue
  22. 61
    36
      src/components/property/propertyPage.vue
  23. 16
    29
      src/components/property/propertySearchFields.vue
  24. 127
    147
      src/components/property/propertySearchPage.vue
  25. 83
    0
      src/components/property/propertySearchResults.vue
  26. 168
    199
      src/components/property/propertyeditPage.vue
  27. 30
    0
      src/components/shared/alertPage.vue
  28. 81
    0
      src/components/shared/autoComplete.vue
  29. 118
    0
      src/components/shared/fieldEditor.vue
  30. 338
    0
      src/components/shared/gallerySlideShow.vue
  31. 51
    11
      src/components/shared/listView.vue
  32. 8
    6
      src/components/shared/navBar.vue
  33. 11
    27
      src/components/shared/searchTab.vue
  34. 13
    3
      src/components/timeshare/buy/buyPage.vue
  35. 38
    10
      src/components/timeshare/myWeeksPage.vue
  36. 9
    0
      src/main.js
  37. 231
    218
      src/router/index.js
  38. 4
    0
      src/store/index.js
  39. 21
    0
      src/store/modules/misc/alert.js
  40. 25
    0
      src/store/modules/misc/carousel.js
  41. 17
    4
      src/store/modules/property/property.js
  42. 11
    8
      src/store/modules/property/propertyEdit.js
  43. 41
    0
      src/store/modules/property/propertySearch.js
  44. 20
    2
      src/store/modules/timeshare/myWeeks.js
  45. 22
    0
      src/store/modules/timeshare/tenderWeeks.js

Binārs
public/img/icons/area.png Parādīt failu


Binārs
public/img/icons/garage.png Parādīt failu


Binārs
public/img/sort-down.png Parādīt failu


Binārs
public/img/sort-up.png Parādīt failu


+ 3
- 0
src/App.vue Parādīt failu

33
 </script>
33
 </script>
34
 
34
 
35
 <style>
35
 <style>
36
+button:hover {
37
+  cursor: pointer;
38
+},
36
 hr {
39
 hr {
37
   background-color: #60cbeb;
40
   background-color: #60cbeb;
38
 }
41
 }

+ 206
- 1
src/components/admin/misc/carousel.vue Parādīt failu

1
-<!-- Work in Progress -->
1
+<template>
2
+  <div>
3
+    <div class="container">
4
+      <div class="container">
5
+        <div class="row">
6
+          <div class="col-md-12 col-lg-8">
7
+            <div class="title-box-d">
8
+              <br />
9
+              <h1 class="title-d" style="text-align:left; font-size: 250%">Carousel Item</h1>
10
+            </div>
11
+            <br />
12
+          </div>
13
+        </div>
14
+      </div>
15
+      <div class="container col-md-12" style="text-align:left">
16
+        <div class="form-goup row">
17
+          <div class="col-md-4">
18
+            <label>Type</label>
19
+            <div class="input-group-prepend">
20
+              <select class="form-control" v-model="selectedType">
21
+                <option value="Timeshare">Timeshare</option>
22
+                <option value="Property">Property</option>
23
+              </select>
24
+              <button
25
+                type="button"
26
+                class="input-group-text fa fa-search"
27
+                style="color: #60CBEB"
28
+                data-toggle="modal"
29
+                data-target="#myModal"
30
+              ></button>
31
+              <!-- Modal content-->
32
+              <div id="myModal" class="modal fade" role="dialog">
33
+                <div class="modal-dialog modal-lg">
34
+                  <div class="modal-content">
35
+                    <div class="modal-header">
36
+                      <button type="button" class="close" data-dismiss="modal"></button>
37
+                    </div>
38
+                    <div padding-left="20px">
39
+                      <Search :name="selectedType" @onSelected="onSelected" />
40
+                    </div>
41
+                    <div class="modal-footer">
42
+                      <button
43
+                        type="button"
44
+                        class="btn btn-b-n"
45
+                        style="width: 150px; height:40px;"
46
+                        data-dismiss="modal"
47
+                      >Close</button>
48
+                    </div>
49
+                  </div>
50
+                </div>
51
+              </div>
52
+              <!-- Modal content END-->
53
+            </div>
54
+          </div>
55
+        </div>
56
+        <div class="form-goup row">
57
+          <div class="col-md-4">
58
+            <br />
59
+            <label>Header</label>
60
+            <input class="form-control" type="text" v-model="carousel.header" />
61
+          </div>
62
+        </div>
63
+        <div class="form-goup row">
64
+          <div class="col-md-4">
65
+            <br />
66
+            <label>Description</label>
67
+            <p v-if="selectedType === 'Property'" v-html="propDescription" />
68
+            <ul v-if="selectedType === 'Timeshare' && bedrooms" class="list">
69
+              <li class="d-flex justify-content-between">
70
+                <strong>
71
+                  <i class="fa fa-bed"></i>
72
+                  &nbsp&nbsp&nbspBedrooms:
73
+                </strong>
74
+                <span>{{ bedrooms }}</span>
75
+              </li>
76
+              <li class="d-flex justify-content-between">
77
+                <strong>
78
+                  <i class="fa fa-users"></i>&nbsp&nbsp&nbspSleeps:
79
+                </strong>
80
+                <span>{{ sleeps }}</span>
81
+              </li>
82
+              <li class="d-flex justify-content-between">
83
+                <strong>
84
+                  <i class="fa fa-calendar"></i>&nbsp&nbsp&nbspCheck in:
85
+                </strong>
86
+                <span>{{ arrival }}</span>
87
+              </li>
88
+              <li class="d-flex justify-content-between">
89
+                <strong>
90
+                  <i class="fa fa-calendar"></i>&nbsp&nbsp&nbspCheck out:
91
+                </strong>
92
+                <span>{{ departure }}</span>
93
+              </li>
94
+            </ul>
95
+          </div>
96
+        </div>
97
+        <div class="form-group row">
98
+          <div class="col-md-2">
99
+            <label>Image</label>
100
+          </div>
101
+        </div>
102
+        <div v-if="selectedType === 'Timeshare'">
103
+          <div>
104
+            <Images :allowMultiple="false" :loadedImages="loadedImages" />
105
+          </div>
106
+        </div>
107
+        <div class="form-group row" v-if="selectedType === 'Property'">
108
+          <div class="col-md-2" v-for="(img, i) in propertyImages" :key="i">
109
+            <input type="radio" name="image" @click="setImage(img)" />
110
+            <label for="checkbox" style="margin: 10px;">Set</label>
111
+            <br />
112
+            <img :src="img.image" style="height:200px; width:150px; object-fit: cover;" />
113
+          </div>
114
+        </div>
115
+        <div class="form-group row">
116
+          <button
117
+            type="button"
118
+            @click="SaveData()"
119
+            class="btn btn-b-n"
120
+            style="width: 85px; height:40px;"
121
+          >Save</button>
122
+          <button
123
+            type="button"
124
+            @click="Close()"
125
+            class="btn btn-b-n"
126
+            style="width: 85px; height:40px;"
127
+          >Close</button>
128
+        </div>
129
+      </div>
130
+    </div>
131
+  </div>
132
+</template>
133
+
134
+<script>
135
+import { mapState, mapActions } from 'vuex';
136
+import Search from './carouselSearch.vue';
137
+import Images from '../../property/propertyImage.vue';
138
+
139
+export default {
140
+  name: 'CarouselItem',
141
+  components: {
142
+    Search,
143
+    Images,
144
+  },
145
+  data() {
146
+    return {
147
+      selectedType: 'Timeshare',
148
+      propDescription: '',
149
+      bedrooms: '',
150
+      sleeps: '',
151
+      arrival: Date,
152
+      departure: Date,
153
+      images: [],
154
+    };
155
+  },
156
+  computed: {
157
+    ...mapState('carousel', ['carousel']),
158
+    ...mapState('property', ['propertyImages']),
159
+  },
160
+  mounted() {
161
+    this.getCarouselItem(this.$route.params.id);
162
+  },
163
+  methods: {
164
+    ...mapActions('carousel', ['saveCarouselItem', 'getCarouselItem']),
165
+    ...mapActions('property', ['getSavedPropertyImages']),
166
+    onSelected(item) {
167
+      if (this.selectedType === 'Timeshare') {
168
+        this.carousel.header = item.resort.resortName;
169
+        this.bedrooms = item.bedrooms;
170
+        this.sleeps = item.maxSleep;
171
+        this.arrival = item.arrivalDate;
172
+        this.departure = item.departureDate;
173
+        this.carousel.timeshareID = item.id;
174
+      } else {
175
+        this.propertyImages = [];
176
+        this.carousel.header = item.name;
177
+        this.propDescription = item.carouselDescription;
178
+        this.carousel.propertyID = item.id;
179
+        this.getSavedPropertyImages(item.id);
180
+      }
181
+    },
182
+    setImage(img) {
183
+      this.carousel.image = img.image;
184
+    },
185
+    loadedImages(values) {
186
+      this.images = values;
187
+    },
188
+    Close() {
189
+      this.$router.push('/carousel');
190
+    },
191
+    SaveData() {
192
+      if (this.selectedType === 'Timeshare') {
193
+        // eslint-disable-next-line no-plusplus
194
+        for (let i = 0; i < this.images.length; i++) {
195
+          this.carousel.image = this.images[i];
196
+        }
197
+      }
198
+      if (this.carousel.id === 0) {
199
+        this.saveCarouselItem(this.carousel);
200
+        this.$router.push('/carousel');
201
+      }
202
+      // Else update
203
+    },
204
+  },
205
+};
206
+</script>

+ 1
- 16
src/components/admin/misc/carouselList.vue Parādīt failu

2
   <!-- eslint-disable max-len -->
2
   <!-- eslint-disable max-len -->
3
   <div>
3
   <div>
4
     <div class="container">
4
     <div class="container">
5
-      <!-- <section class="intro-single"> -->
6
       <div class="container">
5
       <div class="container">
7
-        <br />
8
-        <br />
9
         <div class="row">
6
         <div class="row">
10
           <div class="col-md-12 col-lg-8">
7
           <div class="col-md-12 col-lg-8">
11
-            <!-- <div class="title-single-box"> -->
12
-            <!-- <h1 class="title-single">Property Types</h1> -->
13
-            <!-- </div> -->
14
             <div class="title-box-d">
8
             <div class="title-box-d">
9
+              <br />
15
               <h1 class="title-d" style="text-align:left; font-size: 250%">Carousel Items</h1>
10
               <h1 class="title-d" style="text-align:left; font-size: 250%">Carousel Items</h1>
16
             </div>
11
             </div>
17
             <br />
12
             <br />
18
           </div>
13
           </div>
19
         </div>
14
         </div>
20
       </div>
15
       </div>
21
-      <!-- </section> -->
22
     </div>
16
     </div>
23
     <div class="container">
17
     <div class="container">
24
       <button type="button" @click="New()" class="btn btn-b-n" style="width: 85px; height:40px;">New</button>
18
       <button type="button" @click="New()" class="btn btn-b-n" style="width: 85px; height:40px;">New</button>
31
             <th>Header</th>
25
             <th>Header</th>
32
             <th>Type</th>
26
             <th>Type</th>
33
             <th></th>
27
             <th></th>
34
-            <th></th>
35
           </tr>
28
           </tr>
36
         </thead>
29
         </thead>
37
         <tbody>
30
         <tbody>
42
             <td>{{ item.header }}</td>
35
             <td>{{ item.header }}</td>
43
             <td v-if="item.propertyId">Property</td>
36
             <td v-if="item.propertyId">Property</td>
44
             <td v-else>Timeshare Week</td>
37
             <td v-else>Timeshare Week</td>
45
-            <td>
46
-              <button
47
-                type="button"
48
-                @click="Edit(item.id)"
49
-                class="btn"
50
-                style="margin:2px; color: #60CBEB"
51
-              >Edit</button>
52
-            </td>
53
             <td>
38
             <td>
54
               <button
39
               <button
55
                 type="button"
40
                 type="button"

+ 101
- 0
src/components/admin/misc/carouselSearch.vue Parādīt failu

1
+<template>
2
+  <div>
3
+    <div class="container">
4
+      <div class="container">
5
+        <div class="row">
6
+          <div class="col-md-12 col-lg-8">
7
+            <div class="title-box-d">
8
+              <br />
9
+              <h1 class="title-d" style="text-align:left; font-size: 250%">Search {{ name }}</h1>
10
+            </div>
11
+          </div>
12
+        </div>
13
+        <div class="row">
14
+          <div class="container">
15
+            <ListView
16
+              v-if="name === 'Timeshare'"
17
+              :items="items"
18
+              :showNew="false"
19
+              @onRowClick="onRowClick"
20
+            />
21
+            <ListView v-else :items="properties" :showNew="false" @onRowClick="onRowClick" />
22
+          </div>
23
+        </div>
24
+      </div>
25
+    </div>
26
+  </div>
27
+</template>
28
+
29
+<script>
30
+import { mapState, mapActions } from 'vuex';
31
+import ListView from '../../shared/listView.vue';
32
+import Log from '../../../assets/Log';
33
+
34
+export default {
35
+  name: 'CarouselSearch',
36
+  props: {
37
+    name: String,
38
+  },
39
+  components: {
40
+    ListView,
41
+  },
42
+  data() {
43
+    return {
44
+      user: Log.getUser(),
45
+    };
46
+  },
47
+  mounted() {
48
+    if (this.name === 'Timeshare') {
49
+      this.getItems(this.user.id);
50
+    } else {
51
+      this.getProperties(
52
+        Object.assign(
53
+          {},
54
+          {
55
+            propertyType: 'Admin',
56
+            user: this.user.id,
57
+          },
58
+        ),
59
+      );
60
+    }
61
+  },
62
+  computed: {
63
+    ...mapState('myWeeks', ['items']),
64
+    ...mapState('propertyList', ['properties']),
65
+    // eslint-disable-next-line vue/return-in-computed-property
66
+    nameChanged() {
67
+      if (this.name === 'Timeshare') {
68
+        this.getItems(this.user.id);
69
+      } else {
70
+        this.getProperties(
71
+          Object.assign(
72
+            {},
73
+            {
74
+              propertyType: 'Admin',
75
+              user: this.user.id,
76
+            },
77
+          ),
78
+        );
79
+      }
80
+    },
81
+  },
82
+  methods: {
83
+    ...mapActions('myWeeks', ['getItems']),
84
+    ...mapActions('propertyList', ['getProperties']),
85
+    onRowClick(item) {
86
+      if (this.name === 'Timeshare') {
87
+        const week = this.items[item];
88
+        this.$emit('onSelected', week);
89
+      } else {
90
+        const prop = this.properties[item];
91
+        this.$emit('onSelected', prop);
92
+      }
93
+    },
94
+  },
95
+  watch: {
96
+    nameChanged() {
97
+      return null;
98
+    },
99
+  },
100
+};
101
+</script>

+ 5
- 16
src/components/admin/status/statusPage.vue Parādīt failu

20
     </div>
20
     </div>
21
     <!-- </section> -->
21
     <!-- </section> -->
22
     <div class="container">
22
     <div class="container">
23
-      <table class="table table-bordered">
24
-        <thead>
25
-          <tr>
26
-            <th>Id</th>
27
-            <th>Code</th>
28
-            <th>Description</th>
29
-          </tr>
30
-        </thead>
31
-        <tbody>
32
-          <tr v-for="(item, i) in statusList" :key="i">
33
-            <td>{{item.id}}</td>
34
-            <td>{{item.code}}</td>
35
-            <td>{{item.description}}</td>
36
-          </tr>
37
-        </tbody>
38
-      </table>
23
+      <ListView :items="statusList" :columnCount="4" :compact="false" :showNew="false" />
39
     </div>
24
     </div>
40
   </div>
25
   </div>
41
 </template>
26
 </template>
42
 <script>
27
 <script>
43
 import { mapState, mapActions } from 'vuex';
28
 import { mapState, mapActions } from 'vuex';
29
+import ListView from '../../shared/listView.vue';
44
 
30
 
45
 export default {
31
 export default {
46
   name: 'StatusList',
32
   name: 'StatusList',
33
+  components: {
34
+    ListView,
35
+  },
47
   created() {
36
   created() {
48
     this.getStatusList();
37
     this.getStatusList();
49
   },
38
   },

+ 8
- 4
src/components/admin/status/tenderWeekAdminPage.vue Parādīt failu

17
       </div>
17
       </div>
18
     </div>
18
     </div>
19
     <div id="table" class="col-xs-12 table-responsive">
19
     <div id="table" class="col-xs-12 table-responsive">
20
-      <datatable :columns="columns" :data="rows"></datatable>
20
+      <ListView :items="items" :showNew="false"></listView>
21
     </div>
21
     </div>
22
     <br />
22
     <br />
23
   </div>
23
   </div>
26
 </template>
26
 </template>
27
 <script>
27
 <script>
28
 import { mapState, mapActions } from 'vuex';
28
 import { mapState, mapActions } from 'vuex';
29
+import ListView from '../../shared/listView.vue';
29
 
30
 
30
 export default {
31
 export default {
31
   name: 'TenderWeekAdmin',
32
   name: 'TenderWeekAdmin',
32
   created() {
33
   created() {
33
-    this.gettenderWeekdmin();
34
+    this.getItems();
35
+  },
36
+  components: {
37
+    ListView
34
   },
38
   },
35
   computed: {
39
   computed: {
36
-    ...mapState('status', ['tenderWeekAdmin']),
40
+    ...mapState('tenderWeek', ['items']),
37
   },
41
   },
38
   methods: {
42
   methods: {
39
-    ...mapActions('status', ['gettenderWeekAdmin']),
43
+    ...mapActions('tenderWeek', ['getItems']),
40
   },
44
   },
41
 };
45
 };
42
 </script>
46
 </script>

+ 23
- 11
src/components/admin/status/timeshareAdminPage.vue Parādīt failu

1
+/* eslint-disable no-restricted-syntax */
2
+/* eslint-disable guard-for-in */
1
 <template>
3
 <template>
2
   <div>
4
   <div>
3
-    <br />
4
-    <br />
5
-    <br />
6
-    <br />
7
     <br />
5
     <br />
8
     <div class="row">
6
     <div class="row">
9
       <div class="offset-1 col-md-3">
7
       <div class="offset-1 col-md-3">
11
           type="button"
9
           type="button"
12
           :class="{'form-control btn btn-primary ': (1===1), 'my-disable': (!selectedItems || selectedItems.length === 0)}"
10
           :class="{'form-control btn btn-primary ': (1===1), 'my-disable': (!selectedItems || selectedItems.length === 0)}"
13
           value="Verify Week(s)"
11
           value="Verify Week(s)"
14
-          :disabled="selectedItems.length > 0"
12
+          :disabled="!(selectedItems.length > 0)"
13
+          @click="onVerify()"
15
         >{{ButtonMessage}}</button>
14
         >{{ButtonMessage}}</button>
16
         <div>{{Message}}</div>
15
         <div>{{Message}}</div>
17
       </div>
16
       </div>
18
       <div class="offset-1 col-md-10">
17
       <div class="offset-1 col-md-10">
19
         <ListView
18
         <ListView
20
-          :items="items"
21
-          :allowMultipleSelect="true"
19
+          :items="getNeedsVerify"
22
           @onRowClick="onRowClick"
20
           @onRowClick="onRowClick"
23
           @onClearSelected="onClearSelected"
21
           @onClearSelected="onClearSelected"
24
-          :columnsCount="15"
25
           :title="'Pending Weeks'"
22
           :title="'Pending Weeks'"
23
+          :showNew="false"
26
         />
24
         />
27
       </div>
25
       </div>
28
     </div>
26
     </div>
29
   </div>
27
   </div>
30
 </template>
28
 </template>
31
 <script>
29
 <script>
32
-import { mapState, mapActions } from 'vuex';
30
+import { mapState, mapActions, mapGetters } from 'vuex';
33
 import Log from '../../../assets/Log';
31
 import Log from '../../../assets/Log';
34
 import ListView from '../../shared/listView.vue';
32
 import ListView from '../../shared/listView.vue';
35
 
33
 
46
     };
44
     };
47
   },
45
   },
48
   mounted() {
46
   mounted() {
49
-    console.log(this.user.id);
50
     this.getItems(this.user.id);
47
     this.getItems(this.user.id);
51
   },
48
   },
52
   computed: {
49
   computed: {
81
       return msg;
78
       return msg;
82
     },
79
     },
83
     ...mapState('myWeeks', ['items']),
80
     ...mapState('myWeeks', ['items']),
81
+    ...mapGetters({ getNeedsVerify: 'myWeeks/getNeedsVerify' }),
84
   },
82
   },
85
   methods: {
83
   methods: {
86
-    ...mapActions('myWeeks', ['getItems']),
84
+    ...mapActions('myWeeks', ['getItems', 'verifyWeek']),
85
+    ...mapActions('alert', ['success']),
86
+    onVerify() {
87
+      if (this.selectedItems.length > 0) {
88
+        for (const i in this.selectedItems) {
89
+          const selectedIndex = this.selectedItems[i];
90
+          const item = this.items[selectedIndex];
91
+          if (item) {
92
+            this.verifyWeek(item.id);
93
+          }
94
+        }
95
+        this.success('Week verified!');
96
+        this.$router.push('/shared/alert');
97
+      }
98
+    },
87
     onRowClick(items) {
99
     onRowClick(items) {
88
       this.selectedItems = items;
100
       this.selectedItems = items;
89
     },
101
     },

+ 7
- 3
src/components/communication/templateDetail.vue Parādīt failu

170
         </div>
170
         </div>
171
         <div class="col-md-12">
171
         <div class="col-md-12">
172
           <ListView
172
           <ListView
173
-            :items="item.placeHolders"
173
+            :items="PlaceHolders"
174
             :hideSearch="true"
174
             :hideSearch="true"
175
             :showNew="false"
175
             :showNew="false"
176
             :deleteable="CanEdit"
176
             :deleteable="CanEdit"
200
   },
200
   },
201
   props: {
201
   props: {
202
     item: {},
202
     item: {},
203
+    placeHolders: [],
203
     editable: {
204
     editable: {
204
       default: false,
205
       default: false,
205
     },
206
     },
214
     ItemList() {
215
     ItemList() {
215
       return this.item.placeHolders;
216
       return this.item.placeHolders;
216
     },
217
     },
218
+    PlaceHolders(){
219
+      return this.placeHolders;
220
+    }
217
   },
221
   },
218
   methods: {
222
   methods: {
219
     ...mapActions('template', ['addItem', 'editItem']),
223
     ...mapActions('template', ['addItem', 'editItem']),
230
         // add.
234
         // add.
231
         this.addItem(this.item);
235
         this.addItem(this.item);
232
       }
236
       }
233
-      console.log(JSON.stringify(this.item));
234
       this.$emit('onClose');
237
       this.$emit('onClose');
235
     },
238
     },
236
     onItemDelete(item) {
239
     onItemDelete(item) {
250
         boundToClassDisplay: item.class.name,
253
         boundToClassDisplay: item.class.name,
251
         boundToClass: item.class.fullName,
254
         boundToClass: item.class.fullName,
252
       });
255
       });
253
-      this.item.placeHolders = myList;
256
+      this.placeHolders = myList;
257
+      this.item.placeHolders = this.placeHolders;
254
     },
258
     },
255
   },
259
   },
256
 };
260
 };

+ 1
- 0
src/components/communication/templateInnerItem.vue Parādīt failu

58
       this.getProperties(this.item.class);
58
       this.getProperties(this.item.class);
59
     },
59
     },
60
     onItemAdd() {
60
     onItemAdd() {
61
+      this.message = undefined;
61
       let msg = '';
62
       let msg = '';
62
       if (this.item.name === undefined) {
63
       if (this.item.name === undefined) {
63
         msg += 'Please give a name...';
64
         msg += 'Please give a name...';

+ 10
- 6
src/components/communication/templatePage.vue Parādīt failu

16
         <ListView :items="list" @onRowClick="onRowClick" @onNew="onNew" :showNew="true" />
16
         <ListView :items="list" @onRowClick="onRowClick" @onNew="onNew" :showNew="true" />
17
       </div>
17
       </div>
18
       <div class="col-md-6" v-if="showDetailView">
18
       <div class="col-md-6" v-if="showDetailView">
19
-        <DetailView :item="detailItem" @onClose="onDetailClose" :editable="clickedNew" />
19
+        <DetailView :placeHolders="detailItem ? detailItem.placeHolders : []" :item="detailItem" @onClose="onDetailClose" :editable="clickedNew" />
20
       </div>
20
       </div>
21
     </div>
21
     </div>
22
 
22
 
56
   },
56
   },
57
   methods: {
57
   methods: {
58
     ...mapActions('template', ['getList']),
58
     ...mapActions('template', ['getList']),
59
-    onRowClick(item) {
60
-      if (this.detailItem !== item) {
61
-        this.detailItem = item;
62
-      } else this.detailItem = undefined;
63
-      this.clickedNew = false;
59
+    onRowClick(items) {
60
+      if (items.length > 0) {
61
+        const i = items[0];
62
+        const item = this.list[i];
63
+        if (this.detailItem !== item) {
64
+          this.detailItem = item;
65
+        } else this.detailItem = undefined;
66
+        this.clickedNew = false;
67
+      }
64
     },
68
     },
65
     onNew() {
69
     onNew() {
66
       this.detailItem = {};
70
       this.detailItem = {};

+ 17
- 9
src/components/home/carouselSection.vue Parādīt failu

22
               <div class="col-lg-8">
22
               <div class="col-lg-8">
23
                 <div
23
                 <div
24
                   class="price-a"
24
                   class="price-a"
25
-                  style="opacity:0.95; border: white solid 3px; border-radius: 15px; background-color: white;"
25
+                  style="opacity:0.7; border: white solid 3px; border-radius: 15px; background-color: white;"
26
                 >
26
                 >
27
                   <h1 class="intro-title mb-4">{{ car.header }}</h1>
27
                   <h1 class="intro-title mb-4">{{ car.header }}</h1>
28
-                  <div v-if="car.isProperty">
29
-                    <p class="color-b" v-html="car.address" />
28
+                  <div class="summary-list" v-if="car.isProperty">
29
+                    <p v-html="car.address" />
30
                   </div>
30
                   </div>
31
-                  <div v-else>
32
-                    <ul class="list color-b">
31
+                  <div class="summary-list" v-else>
32
+                    <ul class="list">
33
                       <li class="d-flex justify-content-between">
33
                       <li class="d-flex justify-content-between">
34
-                        <strong>Bedrooms:</strong>
34
+                        <strong>
35
+                          <i class="fa fa-bed"></i>&nbsp&nbsp&nbspBedrooms:
36
+                        </strong>
35
                         <span>{{ car.bedrooms }}</span>
37
                         <span>{{ car.bedrooms }}</span>
36
                       </li>
38
                       </li>
37
                       <li class="d-flex justify-content-between">
39
                       <li class="d-flex justify-content-between">
38
-                        <strong>Sleeps:</strong>
40
+                        <strong>
41
+                          <i class="fa fa-users"></i>&nbsp&nbsp&nbspSleeps:
42
+                        </strong>
39
                         <span>{{ car.sleeps }}</span>
43
                         <span>{{ car.sleeps }}</span>
40
                       </li>
44
                       </li>
41
                       <li class="d-flex justify-content-between">
45
                       <li class="d-flex justify-content-between">
42
-                        <strong>Check in:</strong>
46
+                        <strong>
47
+                          <i class="fa fa-calendar"></i>&nbsp&nbsp&nbspCheck in:
48
+                        </strong>
43
                         <span>{{ car.arrival | toDate }}</span>
49
                         <span>{{ car.arrival | toDate }}</span>
44
                       </li>
50
                       </li>
45
                       <li class="d-flex justify-content-between">
51
                       <li class="d-flex justify-content-between">
46
-                        <strong>Check out:</strong>
52
+                        <strong>
53
+                          <i class="fa fa-calendar"></i>&nbsp&nbsp&nbspCheck out:
54
+                        </strong>
47
                         <span>{{ car.departure | toDate}}</span>
55
                         <span>{{ car.departure | toDate}}</span>
48
                       </li>
56
                       </li>
49
                     </ul>
57
                     </ul>

+ 4
- 0
src/components/processFlow/makeOffer.vue Parādīt failu

196
       if (this.isProperty) {
196
       if (this.isProperty) {
197
         this.bidItem.propertyId = this.item.id;
197
         this.bidItem.propertyId = this.item.id;
198
       } else {
198
       } else {
199
+        this.bidItem.timeshareWeek = this.item;
200
+        this.bidItem.timeshareWeek.id = this.item.resort.id;
201
+        this.bidItem.timeshareWeek.resortCode = this.item.resort.resortCode;
202
+        this.bidItem.timeshareWeek.resortName = this.item.resort.resortName;
199
         this.bidItem.timeshareWeekId = this.item.id;
203
         this.bidItem.timeshareWeekId = this.item.id;
200
       }
204
       }
201
       this.saveBid(this.bidItem);
205
       this.saveBid(this.bidItem);

+ 14
- 4
src/components/property/propertyCard.vue Parādīt failu

71
                 <div class="card-footer-a" v-if="currentProperty.showFooter">
71
                 <div class="card-footer-a" v-if="currentProperty.showFooter">
72
                   <ul class="card-info d-flex justify-content-around">
72
                   <ul class="card-info d-flex justify-content-around">
73
                     <li v-if="currentProperty.area !== null">
73
                     <li v-if="currentProperty.area !== null">
74
-                      <h4 class="card-info-title">Area</h4>
74
+                      <h4 class="card-info-title">
75
+                        <img src="../../../public/img/icons/area.png" height="16px" width="16px" />
76
+                        <!-- <i class="fa fa-ruler-combined"></i> -->
77
+                      </h4>
75
                       <span v-html="currentProperty.area"></span>
78
                       <span v-html="currentProperty.area"></span>
76
                     </li>
79
                     </li>
77
                     <li v-if="currentProperty.beds !== null">
80
                     <li v-if="currentProperty.beds !== null">
78
-                      <h4 class="card-info-title">Beds</h4>
81
+                      <h4 class="card-info-title">
82
+                        <i class="fa fa-bed"></i>
83
+                      </h4>
79
                       <span>{{ currentProperty.beds }}</span>
84
                       <span>{{ currentProperty.beds }}</span>
80
                     </li>
85
                     </li>
81
                     <li v-if="currentProperty.baths !== null">
86
                     <li v-if="currentProperty.baths !== null">
82
-                      <h4 class="card-info-title">Baths</h4>
87
+                      <h4 class="card-info-title">
88
+                        <i class="fa fa-bath"></i>
89
+                      </h4>
83
                       <span>{{ currentProperty.baths }}</span>
90
                       <span>{{ currentProperty.baths }}</span>
84
                     </li>
91
                     </li>
85
                     <li v-if="currentProperty.garages !== null">
92
                     <li v-if="currentProperty.garages !== null">
86
-                      <h4 class="card-info-title">Garages</h4>
93
+                      <h4 class="card-info-title">
94
+                        <!-- <i class="fa fa-warehouse"></i> -->
95
+                        <img src="../../../public/img/icons/garage.png" height="16px" width="16px" />
96
+                      </h4>
87
                       <span>{{ currentProperty.garages }}</span>
97
                       <span>{{ currentProperty.garages }}</span>
88
                     </li>
98
                     </li>
89
                   </ul>
99
                   </ul>

+ 107
- 60
src/components/property/propertyCreate.vue Parādīt failu

46
                     class="form-control"
46
                     class="form-control"
47
                     name="propertyUsageType"
47
                     name="propertyUsageType"
48
                     id="propertyUsageType"
48
                     id="propertyUsageType"
49
-                    v-model="property.propertyUsageType"
49
+                    v-model="propertyType"
50
                     @change="TypeChanged"
50
                     @change="TypeChanged"
51
                   >
51
                   >
52
                     <option value="Residential">Residential</option>
52
                     <option value="Residential">Residential</option>
55
                 </div>
55
                 </div>
56
               </div>
56
               </div>
57
             </div>
57
             </div>
58
+            <br />
58
             <div class="form-group row">
59
             <div class="form-group row">
59
               <div class="col-md-4">
60
               <div class="col-md-4">
60
-                <label>Property Type</label>
61
+                <label>Property Type *</label>
61
                 <div class="input-group-prepend">
62
                 <div class="input-group-prepend">
62
                   <span class="input-group-text" style="color: #60CBEB">
63
                   <span class="input-group-text" style="color: #60CBEB">
63
                     <b>T</b>
64
                     <b>T</b>
67
                     name="propertyType"
68
                     name="propertyType"
68
                     id="propertyType"
69
                     id="propertyType"
69
                     v-model="property.propertyTypeId"
70
                     v-model="property.propertyTypeId"
71
+                    @change="PropertyTypeSelected"
70
                   >
72
                   >
71
                     <option value="0">Please select type</option>
73
                     <option value="0">Please select type</option>
72
-                    <option
73
-                      v-for="item in propertyTypes"
74
-                      :value="item.id"
75
-                      :key="item.id"
76
-                    >{{ item.description }}</option>
74
+                    <option v-for="item in propertyTypes" :value="item.id" :key="item.id">{{
75
+                      item.description
76
+                    }}</option>
77
                   </select>
77
                   </select>
78
                 </div>
78
                 </div>
79
+                <div v-if="showPropertyTypeError">
80
+                  <p class="alert myError">Please select a type</p>
81
+                </div>
79
               </div>
82
               </div>
80
               <div v-if="propertyType === 'Commercial'" class="col-md-4">
83
               <div v-if="propertyType === 'Commercial'" class="col-md-4">
81
                 <label>Property Name</label>
84
                 <label>Property Name</label>
139
                 </div>
142
                 </div>
140
               </div>
143
               </div>
141
               <div class="col-md-6" style="margin-bottom: 1em">
144
               <div class="col-md-6" style="margin-bottom: 1em">
142
-                <label>Province</label>
145
+                <label>Province *</label>
143
                 <div class="input-group-prepend">
146
                 <div class="input-group-prepend">
144
                   <span class="input-group-text">
147
                   <span class="input-group-text">
145
                     <eva-icon name="home-outline" fill="#60CBEB"></eva-icon>
148
                     <eva-icon name="home-outline" fill="#60CBEB"></eva-icon>
152
                     v-model="property.provinceId"
155
                     v-model="property.provinceId"
153
                   >
156
                   >
154
                     <option value="0">Please select province</option>
157
                     <option value="0">Please select province</option>
155
-                    <option
156
-                      v-for="province in provinces"
157
-                      :value="province.id"
158
-                      :key="province.id"
159
-                    >{{ province.description }}</option>
158
+                    <option v-for="province in provinces" :value="province.id" :key="province.id">{{
159
+                      province.description
160
+                    }}</option>
160
                   </select>
161
                   </select>
161
                 </div>
162
                 </div>
163
+                <div v-if="showProvinceError">
164
+                  <p class="alert myError">Please select a Province</p>
165
+                </div>
162
               </div>
166
               </div>
163
 
167
 
164
               <div class="col-md-6" style="margin-bottom: 1em">
168
               <div class="col-md-6" style="margin-bottom: 1em">
165
-                <label>City</label>
169
+                <label>City *</label>
166
                 <div class="input-group-prepend">
170
                 <div class="input-group-prepend">
167
                   <span class="input-group-text">
171
                   <span class="input-group-text">
168
                     <eva-icon name="home-outline" fill="#60CBEB"></eva-icon>
172
                     <eva-icon name="home-outline" fill="#60CBEB"></eva-icon>
175
                     v-model="property.cityId"
179
                     v-model="property.cityId"
176
                   >
180
                   >
177
                     <option value="0">Please select city</option>
181
                     <option value="0">Please select city</option>
178
-                    <option
179
-                      v-for="city in cities"
180
-                      :value="city.id"
181
-                      :key="city.id"
182
-                    >{{ city.description }}</option>
182
+                    <option v-for="city in cities" :value="city.id" :key="city.id">{{
183
+                      city.description
184
+                    }}</option>
183
                   </select>
185
                   </select>
184
                 </div>
186
                 </div>
187
+                <div v-if="showCityError">
188
+                  <p class="alert myError">Please select a City</p>
189
+                </div>
185
               </div>
190
               </div>
186
               <div class="col-md-6" style="margin-bottom: 1em">
191
               <div class="col-md-6" style="margin-bottom: 1em">
187
-                <label>Suburb</label>
192
+                <label>Suburb *</label>
188
                 <div class="input-group-prepend">
193
                 <div class="input-group-prepend">
189
                   <span class="input-group-text">
194
                   <span class="input-group-text">
190
                     <eva-icon name="home-outline" fill="#60CBEB"></eva-icon>
195
                     <eva-icon name="home-outline" fill="#60CBEB"></eva-icon>
197
                     @change="getPostalCode"
202
                     @change="getPostalCode"
198
                   >
203
                   >
199
                     <option value="0">Please select suburb</option>
204
                     <option value="0">Please select suburb</option>
200
-                    <option
201
-                      v-for="suburb in suburbs"
202
-                      :value="suburb.id"
203
-                      :key="suburb.id"
204
-                    >{{ suburb.description }}</option>
205
+                    <option v-for="suburb in suburbs" :value="suburb.id" :key="suburb.id">{{
206
+                      suburb.description
207
+                    }}</option>
205
                   </select>
208
                   </select>
206
                 </div>
209
                 </div>
210
+                <div v-if="showSuburbError">
211
+                  <p class="alert myError">Please select a Suburb</p>
212
+                </div>
207
               </div>
213
               </div>
208
               <div class="col-md-6" style="margin-bottom: 1em">
214
               <div class="col-md-6" style="margin-bottom: 1em">
209
                 <label>Postal Code</label>
215
                 <label>Postal Code</label>
263
                 <label for="Property Description">Description</label>
269
                 <label for="Property Description">Description</label>
264
                 <vue-editor v-model="property.description" :editor-toolbar="customToolbar" />
270
                 <vue-editor v-model="property.description" :editor-toolbar="customToolbar" />
265
                 <br />
271
                 <br />
266
-                <p>* A listing fee of R380 including VAT is payable to list your Property on the Uni-Vate website</p>
272
+                <p>
273
+                  * A listing fee of R380 including VAT is payable to list your Property on the
274
+                  Uni-Vate website
275
+                </p>
267
               </div>
276
               </div>
268
             </div>
277
             </div>
269
             <div class="form-group row" />
278
             <div class="form-group row" />
270
             <UserField
279
             <UserField
271
-              v-if="propertyType === 'Residential' & propertyOverviewFields.length > 0"
280
+              v-if="(propertyType === 'Residential') & (propertyOverviewFields.length > 0)"
272
               :fields="propertyOverviewFields[0].fields"
281
               :fields="propertyOverviewFields[0].fields"
273
               @UpdateUserDefinedFields="UpdateUserDefinedFields"
282
               @UpdateUserDefinedFields="UpdateUserDefinedFields"
274
               :id="1"
283
               :id="1"
348
               @click="SubmitData()"
357
               @click="SubmitData()"
349
               class="btn btn-b-n"
358
               class="btn btn-b-n"
350
               style="width: 85px; height:40px;"
359
               style="width: 85px; height:40px;"
351
-            >Save</button>
360
+            >
361
+              Save
362
+            </button>
363
+            <div
364
+              v-if="showPropertyTypeError || showProvinceError || showCityError || showSuburbError"
365
+            >
366
+              <p class="alert myError">
367
+                Missing fields. Please fill in all required fields. Marked with *
368
+              </p>
369
+            </div>
352
             <div v-if="wait" id="preloader"></div>
370
             <div v-if="wait" id="preloader"></div>
353
           </form>
371
           </form>
354
         </div>
372
         </div>
360
 <script>
378
 <script>
361
 import { mapState, mapActions } from 'vuex';
379
 import { mapState, mapActions } from 'vuex';
362
 import { VueEditor } from 'vue2-editor';
380
 import { VueEditor } from 'vue2-editor';
363
-import { setTimeout } from 'timers';
364
 import UserField from './propertyUserField.vue';
381
 import UserField from './propertyUserField.vue';
365
 import ImageLoad from './propertyImage.vue';
382
 import ImageLoad from './propertyImage.vue';
366
 
383
 
384
       customToolbar: [
401
       customToolbar: [
385
         [{ header: [false, 1, 2, 3, 4, 5, 6] }],
402
         [{ header: [false, 1, 2, 3, 4, 5, 6] }],
386
         ['bold', 'italic', 'underline', 'strike'],
403
         ['bold', 'italic', 'underline', 'strike'],
387
-        [
388
-          { align: '' },
389
-          { align: 'center' },
390
-          { align: 'right' },
391
-          { align: 'justify' },
392
-        ],
404
+        [{ align: '' }, { align: 'center' }, { align: 'right' }, { align: 'justify' }],
393
         [{ list: 'ordered' }, { list: 'bullet' }, { list: 'check' }],
405
         [{ list: 'ordered' }, { list: 'bullet' }, { list: 'check' }],
394
         [{ script: 'sub' }, { script: 'super' }],
406
         [{ script: 'sub' }, { script: 'super' }],
395
         [{ indent: '-1' }, { indent: '+1' }],
407
         [{ indent: '-1' }, { indent: '+1' }],
396
       ],
408
       ],
409
+      error: '',
410
+      showPropertyTypeError: false,
411
+      showProvinceError: false,
412
+      showCityError: false,
413
+      showSuburbError: false,
397
     };
414
     };
398
   },
415
   },
399
   methods: {
416
   methods: {
416
       this.propertyType = this.property.propertyUsageType;
433
       this.propertyType = this.property.propertyUsageType;
417
     },
434
     },
418
     SubmitData() {
435
     SubmitData() {
436
+      if (this.property.propertyTypeId === 0) {
437
+        this.showPropertyTypeError = true;
438
+      }
439
+      if (this.property.provinceId === 0) {
440
+        this.showProvinceError = true;
441
+      }
442
+      if (this.property.cityId === 0) {
443
+        this.showCityError = true;
444
+      }
445
+      if (this.property.suburbId === 0) {
446
+        this.showSuburbError = true;
447
+      }
448
+
449
+      if (
450
+        this.showPropertyTypeError
451
+        || this.showProvinceError
452
+        || this.showCityError
453
+        || this.showSuburbError
454
+      ) {
455
+        return;
456
+      }
457
+
419
       this.wait = true;
458
       this.wait = true;
420
       if (this.salesType === 'Sale') {
459
       if (this.salesType === 'Sale') {
421
         this.property.isSale = true;
460
         this.property.isSale = true;
433
       }
472
       }
434
       this.property.propertyUserFields = this.propertyFieldValues;
473
       this.property.propertyUserFields = this.propertyFieldValues;
435
 
474
 
436
-      this.property.userId = this.user.id;
437
-      this.saveProperty(this.property);
438
-      setTimeout(
439
-        () => this.$router.push({
440
-            path: '/property/Search',
441
-            query: {
442
-              salesType: this.salesType,
443
-              propertyUsageType: this.propertyType,
444
-            },
445
-          }),
446
-        3000,
447
-      );
475
+      if (this.user) {
476
+        this.property.userId = this.user.id;
477
+      }
478
+
479
+      this.saveProperty(this.property)
480
+        .then((fulfilled) => {
481
+          this.$router.push(`/property/property/${fulfilled.data.id}`);
482
+        })
483
+        .catch((error) => {
484
+          console.log(error.message);
485
+        });
448
     },
486
     },
449
     Close() {
487
     Close() {
450
       this.$router.push('/property/admin/list/my');
488
       this.$router.push('/property/admin/list/my');
451
     },
489
     },
490
+    PropertyTypeSelected(item) {
491
+      if (item.target.options.selectedIndex > 0) {
492
+        this.showPropertyTypeError = false;
493
+      } else {
494
+        this.showPropertyTypeError = true;
495
+      }
496
+    },
452
     ProvinceSelected(item) {
497
     ProvinceSelected(item) {
453
       if (item.target.options.selectedIndex > 0) {
498
       if (item.target.options.selectedIndex > 0) {
454
-        this.selectedProvince = this.provinces[
455
-          item.target.options.selectedIndex - 1
456
-        ].description;
499
+        this.selectedProvince = this.provinces[item.target.options.selectedIndex - 1].description;
457
         this.getCities(Object.assign({}, { province: this.selectedProvince }));
500
         this.getCities(Object.assign({}, { province: this.selectedProvince }));
501
+        this.showProvinceError = false;
502
+      } else {
503
+        this.showProvinceError = true;
458
       }
504
       }
459
     },
505
     },
460
     CitySelected(item) {
506
     CitySelected(item) {
461
       if (item.target.options.selectedIndex > 0) {
507
       if (item.target.options.selectedIndex > 0) {
462
-        this.selectedCity = this.cities[
463
-          item.target.options.selectedIndex - 1
464
-        ].description;
508
+        this.selectedCity = this.cities[item.target.options.selectedIndex - 1].description;
465
         this.getSuburbs(
509
         this.getSuburbs(
466
-          Object.assign(
467
-            {},
468
-            { province: this.selectedProvince, city: this.selectedCity },
469
-          ),
510
+          Object.assign({}, { province: this.selectedProvince, city: this.selectedCity }),
470
         );
511
         );
512
+        this.showCityError = false;
513
+      } else {
514
+        this.showCityError = true;
471
       }
515
       }
472
     },
516
     },
473
     getPostalCode(item) {
517
     getPostalCode(item) {
474
-      this.property.addressLine3 = this.suburbs[
475
-        item.target.options.selectedIndex - 1
476
-      ].postalCode;
518
+      this.property.addressLine3 = this.suburbs[item.target.options.selectedIndex - 1].postalCode;
519
+      if (item.target.options.selectedIndex > 0) {
520
+        this.showSuburbError = false;
521
+      } else {
522
+        this.showSuburbError = true;
523
+      }
477
     },
524
     },
478
     loadedImages(values) {
525
     loadedImages(values) {
479
       this.images = values;
526
       this.images = values;

+ 12
- 7
src/components/property/propertyFieldEditor.vue Parādīt failu

11
           v-model="display"
11
           v-model="display"
12
           disabled
12
           disabled
13
         />
13
         />
14
-        <span v-if="mayEdit" @click="EditClick()" class="input-group-text" style="color: #60CBEB">
14
+        <span
15
+          v-if="mayEdit"
16
+          @click="EditClick()"
17
+          class="input-group-text spanCursor"
18
+          style="color: #60CBEB"
19
+        >
15
           <eva-icon name="edit-outline" fill="#60CBEB"></eva-icon>
20
           <eva-icon name="edit-outline" fill="#60CBEB"></eva-icon>
16
         </span>
21
         </span>
17
       </div>
22
       </div>
42
           @change="SelectorSelected"
47
           @change="SelectorSelected"
43
         >
48
         >
44
           <option value="0"></option>
49
           <option value="0"></option>
45
-          <option v-for="item in arrayObject" :value="item.id" :key="item.id">{{ item.description }}</option>
50
+          <option v-for="item in arrayObject" :value="item.id" :key="item.id">{{
51
+            item.description
52
+          }}</option>
46
         </select>
53
         </select>
47
         <!-- yesno -->
54
         <!-- yesno -->
48
         <select v-if="editType === 'yesno'" class="form-control" @change="YesNoSelected">
55
         <select v-if="editType === 'yesno'" class="form-control" @change="YesNoSelected">
50
           <option value="yes">Yes</option>
57
           <option value="yes">Yes</option>
51
           <option value="no">No</option>
58
           <option value="no">No</option>
52
         </select>
59
         </select>
53
-        <span @click="UpdateValue()" class="input-group-text" style="color: #60CBEB">
60
+        <span @click="UpdateValue()" class="input-group-text spanCursor" style="color: #60CBEB">
54
           <eva-icon name="checkmark-outline" fill="#60CBEB"></eva-icon>
61
           <eva-icon name="checkmark-outline" fill="#60CBEB"></eva-icon>
55
         </span>
62
         </span>
56
-        <span @click="Close()" class="input-group-text" style="color: #60CBEB">
63
+        <span @click="Close()" class="input-group-text spanCursor" style="color: #60CBEB">
57
           <eva-icon name="close-outline" fill="#60CBEB"></eva-icon>
64
           <eva-icon name="close-outline" fill="#60CBEB"></eva-icon>
58
         </span>
65
         </span>
59
       </div>
66
       </div>
94
     },
101
     },
95
     SelectorSelected(item) {
102
     SelectorSelected(item) {
96
       if (item.target.options.selectedIndex > 0) {
103
       if (item.target.options.selectedIndex > 0) {
97
-        this.updatedDisplay = this.arrayObject[
98
-          item.target.options.selectedIndex - 1
99
-        ].description;
104
+        this.updatedDisplay = this.arrayObject[item.target.options.selectedIndex - 1].description;
100
       }
105
       }
101
     },
106
     },
102
     YesNoSelected(item) {
107
     YesNoSelected(item) {

+ 16
- 4
src/components/property/propertyImage.vue Parādīt failu

10
           style="width: 0px;height: 0px;overflow: hidden;"
10
           style="width: 0px;height: 0px;overflow: hidden;"
11
           name="images[]"
11
           name="images[]"
12
           @change="imagesAdd"
12
           @change="imagesAdd"
13
-          multiple
13
+          :multiple="allowMultiple"
14
           :disabled="!mayEdit"
14
           :disabled="!mayEdit"
15
         />
15
         />
16
       </label>
16
       </label>
18
     <br />
18
     <br />
19
     <div class="form-group row">
19
     <div class="form-group row">
20
       <div v-for="(img, i) in image" class="col-md-2" :key="i">
20
       <div v-for="(img, i) in image" class="col-md-2" :key="i">
21
-        <input type="checkbox" id="checkbox" v-model="imagesDefault[i]" @change="updateList(i)" />
22
-        <label for="checkbox" style="margin: 10px;">Main Image</label>
21
+        <input
22
+          v-if="allowMultiple"
23
+          type="checkbox"
24
+          id="checkbox"
25
+          v-model="imagesDefault[i]"
26
+          @change="updateList(i)"
27
+        />
28
+        <label v-if="allowMultiple" for="checkbox" style="margin: 10px;">Main Image</label>
23
         <img :src="img" style="height:200px; width:150px; object-fit: cover;" />
29
         <img :src="img" style="height:200px; width:150px; object-fit: cover;" />
24
         <br />
30
         <br />
25
-        <span class="input-group-text" align="center" @click="removeImage(key)">
31
+        <span class="input-group-text" align="center" style="width:150px" @click="removeImage(key)">
26
           <eva-icon name="trash-2-outline" fill="#60CBEB"></eva-icon>Delete
32
           <eva-icon name="trash-2-outline" fill="#60CBEB"></eva-icon>Delete
27
         </span>
33
         </span>
28
       </div>
34
       </div>
36
   props: {
42
   props: {
37
     loadedImages: Function,
43
     loadedImages: Function,
38
     mayEdit: { type: Boolean, default: () => true },
44
     mayEdit: { type: Boolean, default: () => true },
45
+    allowMultiple: { type: Boolean, default: () => true },
39
   },
46
   },
40
   data() {
47
   data() {
41
     return {
48
     return {
54
   // },
61
   // },
55
   methods: {
62
   methods: {
56
     imagesAdd(e) {
63
     imagesAdd(e) {
64
+      if (!this.allowMultiple) {
65
+        this.images = {};
66
+        this.image = [];
67
+      }
68
+
57
       const files = e.target.files || e.dataTransfer.files;
69
       const files = e.target.files || e.dataTransfer.files;
58
 
70
 
59
       this.images = [];
71
       this.images = [];

+ 33
- 84
src/components/property/propertyList.vue Parādīt failu

2
   <!-- eslint-disable max-len -->
2
   <!-- eslint-disable max-len -->
3
   <div>
3
   <div>
4
     <div class="container">
4
     <div class="container">
5
-      <!-- <section class="intro-single"> -->
6
       <div class="container">
5
       <div class="container">
7
-        <br />
8
-        <br />
9
         <br />
6
         <br />
10
         <br />
7
         <br />
11
         <div class="row">
8
         <div class="row">
12
           <div class="col-md-12 col-lg-8">
9
           <div class="col-md-12 col-lg-8">
13
             <div class="title-box-d">
10
             <div class="title-box-d">
14
-              <h1
15
-                v-if="showAdmin"
16
-                class="title-d"
17
-                style="text-align:left; font-size: 250%"
18
-              >Admin Properties</h1>
19
-              <h1 v-else class="title-d" style="text-align:left; font-size: 250%">My Properties</h1>
11
+              <h1 class="title-d" style="text-align:left; font-size: 250%">Properties</h1>
20
             </div>
12
             </div>
21
           </div>
13
           </div>
22
         </div>
14
         </div>
23
       </div>
15
       </div>
24
-      <!-- </section> -->
25
     </div>
16
     </div>
26
     <div class="container">
17
     <div class="container">
27
-      <table class="table table-bordered">
28
-        <thead>
29
-          <tr>
30
-            <th>Name</th>
31
-            <th>Property ID</th>
32
-            <th>Size</th>
33
-            <th>Price</th>
34
-            <th>Usage Type</th>
35
-            <th>Type</th>
36
-            <th>Sale Type</th>
37
-            <th>Publish</th>
38
-            <th>Status</th>
39
-            <th></th>
40
-            <th></th>
41
-          </tr>
42
-        </thead>
43
-        <tbody>
44
-          <tr v-for="(item, i) in properties" :key="i">
45
-            <td>{{ item.name }}</td>
46
-            <td>{{ item.id }}</td>
47
-            <td v-html="item.size" />
48
-            <td>{{ item.price }}</td>
49
-            <td>{{ item.usageType }}</td>
50
-            <td>{{ item.type }}</td>
51
-            <td>{{ item.saleType }}</td>
52
-            <!-- <td>{{ item.publish }}</td> -->
53
-            <td></td>
54
-            <td>{{ item.status }}</td>
55
-            <td>
56
-              <button
57
-                type="button"
58
-                @click="Edit(item)"
59
-                class="btn"
60
-                style="margin:2px; color: #60CBEB"
61
-              >Edit</button>
62
-            </td>
63
-            <td>
64
-              <button
65
-                type="button"
66
-                @click="Delete(item)"
67
-                class="btn btn-b-n"
68
-                style="width: 85px; height:40px;"
69
-              >Delete</button>
70
-            </td>
71
-          </tr>
72
-        </tbody>
73
-      </table>
18
+      <listView
19
+        :items="properties"
20
+        :showNew="false"
21
+        :editable="true"
22
+        :deleteable="true"
23
+        @onEdit="Edit"
24
+        @onDelete="Delete"
25
+      />
74
     </div>
26
     </div>
75
     <br />
27
     <br />
76
   </div>
28
   </div>
78
 
30
 
79
 <script>
31
 <script>
80
 import { mapState, mapActions } from 'vuex';
32
 import { mapState, mapActions } from 'vuex';
33
+import listView from '../shared/listView.vue';
81
 
34
 
82
 export default {
35
 export default {
83
   name: 'PropertyList',
36
   name: 'PropertyList',
37
+  components: {
38
+    listView,
39
+  },
84
   data() {
40
   data() {
85
     return {
41
     return {
86
       propertyType: '',
42
       propertyType: '',
87
-      showAdmin: false,
43
+      role: 'MY',
88
     };
44
     };
89
   },
45
   },
90
   methods: {
46
   methods: {
101
     },
57
     },
102
   },
58
   },
103
   mounted() {
59
   mounted() {
104
-    if (this.$route.params.by === 'my') {
105
-      // eslint-disable-next-line vue/no-side-effects-in-computed-properties
106
-      this.showAdmin = false;
107
-    } else {
108
-      // eslint-disable-next-line vue/no-side-effects-in-computed-properties
109
-      this.showAdmin = true;
60
+    if (this.user.role === 'Super Admin') {
61
+      this.role = 'SUPERADMIN';
62
+    }
63
+    if (this.user.role === 'Agency') {
64
+      this.user.role = 'ADMIN';
110
     }
65
     }
111
-
112
-    // this.user = 28; // user id
113
 
66
 
114
     this.getProperties(
67
     this.getProperties(
115
-      Object.assign(
116
-        {},
117
-        {
118
-          propertyType: this.showAdmin ? 'Admin' : 'My',
119
-          user: this.user.id,
120
-        },
121
-      ),
68
+      Object.assign({
69
+        propertyType: this.role,
70
+        user: this.user.id,
71
+      }),
122
     );
72
     );
123
   },
73
   },
124
   computed: {
74
   computed: {
125
     ...mapState('propertyList', ['properties']),
75
     ...mapState('propertyList', ['properties']),
126
     ...mapState('authentication', ['user']),
76
     ...mapState('authentication', ['user']),
127
     UserChanged() {
77
     UserChanged() {
128
-      if (this.$route.params.by === 'my') {
78
+      if (this.user.role === 'Super Admin') {
129
         // eslint-disable-next-line vue/no-side-effects-in-computed-properties
79
         // eslint-disable-next-line vue/no-side-effects-in-computed-properties
130
-        this.showAdmin = false;
131
-      } else {
80
+        this.role = 'SUPERADMIN';
81
+      }
82
+      if (this.user.role === 'Agency') {
132
         // eslint-disable-next-line vue/no-side-effects-in-computed-properties
83
         // eslint-disable-next-line vue/no-side-effects-in-computed-properties
133
-        this.showAdmin = true;
84
+        this.user.role = 'ADMIN';
134
       }
85
       }
86
+
135
       this.getProperties(
87
       this.getProperties(
136
-        Object.assign(
137
-          {},
138
-          {
139
-            propertyType: this.showAdmin ? 'Admin' : 'My',
140
-            user: this.user,
141
-          },
142
-        ),
88
+        Object.assign({
89
+          propertyType: this.role,
90
+          user: this.user.id,
91
+        }),
143
       );
92
       );
144
       return this.user;
93
       return this.user;
145
     },
94
     },

+ 61
- 36
src/components/property/propertyPage.vue Parādīt failu

2
   <!-- eslint-disable max-len -->
2
   <!-- eslint-disable max-len -->
3
   <div v-if="property">
3
   <div v-if="property">
4
     <div class="container">
4
     <div class="container">
5
-      <br />
6
-      <br />
7
-      <br />
8
       <br />
5
       <br />
9
       <div class="row">
6
       <div class="row">
10
         <div class="col-md-12 col-lg-8">
7
         <div class="col-md-12 col-lg-8">
11
           <div class="title-box-d">
8
           <div class="title-box-d">
12
-            <h1
13
-              class="title-d"
14
-              style="text-align:left; font-size: 250%"
15
-            >{{ property.shortDescription }}</h1>
9
+            <h1 class="title-d" style="text-align:left; font-size: 250%">
10
+              {{ property.shortDescription }}
11
+            </h1>
16
           </div>
12
           </div>
17
         </div>
13
         </div>
18
       </div>
14
       </div>
23
     <section class="property-single nav-arrow-b">
19
     <section class="property-single nav-arrow-b">
24
       <div class="container">
20
       <div class="container">
25
         <div class="row">
21
         <div class="row">
26
-          <lightBox
27
-            :thumbnails="propertyImages"
28
-            :largeImages="propertyImages"
29
-            :caption="false"
30
-            class="lightBox"
31
-          />
22
+          <div id="app" class="scrolling-wrapper">
23
+            <div class="card">
24
+              <img
25
+                class="image"
26
+                v-for="(image, i) in propertyImages"
27
+                :src="image"
28
+                @click="index = i"
29
+                :key="i"
30
+              />
31
+            </div>
32
+            <gallery :images="propertyImages" :index="index" @close="index = null"></gallery>
33
+          </div>
32
         </div>
34
         </div>
35
+        <br />
33
         <div class="row">
36
         <div class="row">
34
           <div class="col-sm-12">
37
           <div class="col-sm-12">
35
             <div class="row justify-content-between">
38
             <div class="row justify-content-between">
72
 
75
 
73
                     <div class="card-title-c align-self-center">
76
                     <div class="card-title-c align-self-center">
74
                       <h5 class="title-c">R{{ formatPrice(property.price) }}</h5>
77
                       <h5 class="title-c">R{{ formatPrice(property.price) }}</h5>
75
-                      <h6 v-if="property.pricePer">Per {{property.pricePer}}</h6>
78
+                      <h6 v-if="property.pricePer">Per {{ property.pricePer }}</h6>
76
                     </div>
79
                     </div>
77
                   </div>
80
                   </div>
78
                   <!-- </div> -->
81
                   <!-- </div> -->
93
                       </li>
96
                       </li>
94
                       <li class="d-flex justify-content-between">
97
                       <li class="d-flex justify-content-between">
95
                         <strong>Status:</strong>
98
                         <strong>Status:</strong>
96
-                        <span
97
-                          v-if="property.status"
98
-                        >{{ property.status.code }} - {{ property.status.description }}</span>
99
+                        <span v-if="property.status"
100
+                          >{{ property.status.code }} - {{ property.status.description }}</span
101
+                        >
99
                       </li>
102
                       </li>
100
                       <li class="d-flex justify-content-between">
103
                       <li class="d-flex justify-content-between">
101
                         <strong>Address:</strong>
104
                         <strong>Address:</strong>
102
                         <span
105
                         <span
103
                           style="text-align:right"
106
                           style="text-align:right"
104
                           v-if="property"
107
                           v-if="property"
105
-                          v-html="formatAddress(property.addressLine1) + formatAddress(property.addressLine2) + formatAddress(property.addressLine3) + formatAddress(property.suburb ? property.suburb.description : '') + formatAddress(property.city ? property.city.description : '') + formatAddress(property.province ? property.province.description : '') "
108
+                          v-html="
109
+                            formatAddress(property.addressLine1) +
110
+                              formatAddress(property.addressLine2) +
111
+                              formatAddress(property.addressLine3) +
112
+                              formatAddress(property.suburb ? property.suburb.description : '') +
113
+                              formatAddress(property.city ? property.city.description : '') +
114
+                              formatAddress(property.province ? property.province.description : '')
115
+                          "
106
                         ></span>
116
                         ></span>
107
                       </li>
117
                       </li>
108
                     </ul>
118
                     </ul>
114
                     class="btn btn-b-n"
124
                     class="btn btn-b-n"
115
                     data-toggle="modal"
125
                     data-toggle="modal"
116
                     data-target="#myModal"
126
                     data-target="#myModal"
117
-                  >Make an Offer</button>
127
+                  >
128
+                    Make an Offer
129
+                  </button>
118
                   <div id="myModal" class="modal fade" role="dialog">
130
                   <div id="myModal" class="modal fade" role="dialog">
119
                     <div class="modal-dialog modal-lg">
131
                     <div class="modal-dialog modal-lg">
120
                       <!-- Modal content-->
132
                       <!-- Modal content-->
127
                             name="MakeOffer"
139
                             name="MakeOffer"
128
                             :isMakeOffer="true"
140
                             :isMakeOffer="true"
129
                             :isProperty="true"
141
                             :isProperty="true"
130
-                            :item="{id: property.id, shortDescription: property.shortDescription, description: property.description, sellPrice: property.price}"
142
+                            :item="{
143
+                              id: property.id,
144
+                              shortDescription: property.shortDescription,
145
+                              description: property.description,
146
+                              sellPrice: property.price
147
+                            }"
131
                           />
148
                           />
132
                         </div>
149
                         </div>
133
                       </div>
150
                       </div>
252
 
269
 
253
 <script>
270
 <script>
254
 import { mapState, mapActions } from 'vuex';
271
 import { mapState, mapActions } from 'vuex';
255
-import lightBox from '../shared/lightBoxGallery.vue';
256
 import makeOffer from '../processFlow/makeOffer.vue';
272
 import makeOffer from '../processFlow/makeOffer.vue';
273
+import gallery from '../shared/gallerySlideShow.vue';
257
 
274
 
258
 export default {
275
 export default {
259
   name: 'property',
276
   name: 'property',
260
   components: {
277
   components: {
261
-    lightBox,
262
     makeOffer,
278
     makeOffer,
279
+    gallery,
263
   },
280
   },
264
   data() {
281
   data() {
265
-    return {};
282
+    return {
283
+      index: null,
284
+    };
266
   },
285
   },
267
   mounted() {
286
   mounted() {
268
     this.getProperty(this.$route.params.id);
287
     this.getProperty(this.$route.params.id);
272
     ...mapState('property', ['property', 'propertyImages']),
291
     ...mapState('property', ['property', 'propertyImages']),
273
   },
292
   },
274
   methods: {
293
   methods: {
275
-    ...mapActions('property', [
276
-      'getProperty',
277
-      'getPropertyImages',
278
-      'clearPropertyImages',
279
-    ]),
294
+    ...mapActions('property', ['getProperty', 'getPropertyImages', 'clearPropertyImages']),
280
     formatPrice(value) {
295
     formatPrice(value) {
281
       const val = (value / 1).toFixed(2);
296
       const val = (value / 1).toFixed(2);
282
       return val.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ' ');
297
       return val.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ' ');
294
 };
309
 };
295
 </script>
310
 </script>
296
 
311
 
297
-<style lang ="scss">
298
-.light-box {
299
-  &__thumbnail {
300
-    margin: 20px;
301
-    width: 200px;
302
-  }
312
+<style lang="scss">
313
+.image {
314
+  width: 150px;
315
+  height: 150px;
316
+  background-size: cover;
317
+  cursor: pointer;
318
+  margin: 5px;
319
+  border-radius: 3px;
320
+  border: 1px solid lightgray;
321
+  object-fit: contain;
303
 }
322
 }
304
-img {
305
-  max-width: 100%;
323
+.scrolling-wrapper {
324
+  overflow-x: scroll;
325
+  overflow-y: hidden;
326
+  white-space: nowrap;
327
+
328
+  .card {
329
+    display: inline-block;
330
+  }
306
 }
331
 }
307
 </style>
332
 </style>

+ 16
- 29
src/components/property/propertySearchFields.vue Parādīt failu

9
               <b>F</b>
9
               <b>F</b>
10
             </span>
10
             </span>
11
           </div>
11
           </div>
12
-          <select
13
-            class="form-control"
14
-            v-model="propertySearch.salesType"
15
-            @change="salesTypeSelected"
16
-          >
12
+          <select class="form-control" v-model="propertySearch.salesType">
17
             <option value="Sale">Sale</option>
13
             <option value="Sale">Sale</option>
18
             <option value="Rent">Rent</option>
14
             <option value="Rent">Rent</option>
19
           </select>
15
           </select>
29
           </div>
25
           </div>
30
           <select class="form-control" v-model="resType" @change="PropertyTypeSelected">
26
           <select class="form-control" v-model="resType" @change="PropertyTypeSelected">
31
             <option>All</option>
27
             <option>All</option>
32
-            <option
33
-              v-for="(propertyType, i) in propertyTypesRes"
34
-              :key="i"
35
-            >{{ propertyType.description }}</option>
28
+            <option v-for="(propertyType, i) in propertyTypesRes" :key="i">
29
+              {{
30
+              propertyType.description
31
+              }}
32
+            </option>
36
           </select>
33
           </select>
37
           <div class="input-group-append" @click="clearResType">
34
           <div class="input-group-append" @click="clearResType">
38
             <span class="input-group-text cursor-pointer" style="color: #60CBEB">
35
             <span class="input-group-text cursor-pointer" style="color: #60CBEB">
51
           </div>
48
           </div>
52
           <select class="form-control" v-model="comType" @change="PropertyTypeSelected">
49
           <select class="form-control" v-model="comType" @change="PropertyTypeSelected">
53
             <option>All</option>
50
             <option>All</option>
54
-            <option
55
-              v-for="(propertyType, i) in propertyTypesCom"
56
-              :key="i"
57
-            >{{ propertyType.description }}</option>
51
+            <option v-for="(propertyType, i) in propertyTypesCom" :key="i">
52
+              {{
53
+              propertyType.description
54
+              }}
55
+            </option>
58
           </select>
56
           </select>
59
           <div class="input-group-append" @click="clearComType">
57
           <div class="input-group-append" @click="clearComType">
60
             <span class="input-group-text cursor-pointer" style="color: #60CBEB">
58
             <span class="input-group-text cursor-pointer" style="color: #60CBEB">
71
               <b>P</b>
69
               <b>P</b>
72
             </span>
70
             </span>
73
           </div>
71
           </div>
74
-          <select class="form-control" @change="ProvinceSelected" v-model="propertySearch.province">
72
+          <select class="form-control" v-model="propertySearch.province">
75
             <option>All</option>
73
             <option>All</option>
76
             <option v-for="(province, i) in provinces" :key="i">{{ province.description }}</option>
74
             <option v-for="(province, i) in provinces" :key="i">{{ province.description }}</option>
77
           </select>
75
           </select>
92
               <b>C</b>
90
               <b>C</b>
93
             </span>
91
             </span>
94
           </div>
92
           </div>
95
-          <select class="form-control" @change="CitySelected" v-model="propertySearch.city">
93
+          <select class="form-control" v-model="propertySearch.city">
96
             <option>All</option>
94
             <option>All</option>
97
             <option v-for="(city, i) in cities" :key="i">{{ city.description }}</option>
95
             <option v-for="(city, i) in cities" :key="i">{{ city.description }}</option>
98
           </select>
96
           </select>
113
               <b>S</b>
111
               <b>S</b>
114
             </span>
112
             </span>
115
           </div>
113
           </div>
116
-          <select class="form-control" v-model="propertySearch.suburb" @change="SuburbSeleted">
114
+          <select class="form-control" v-model="propertySearch.suburb">
117
             <option>All</option>
115
             <option>All</option>
118
             <option v-for="(suburb, i) in suburbs" :key="i">{{ suburb.description }}</option>
116
             <option v-for="(suburb, i) in suburbs" :key="i">{{ suburb.description }}</option>
119
           </select>
117
           </select>
139
               <input
137
               <input
140
                 class="form-control"
138
                 class="form-control"
141
                 type="number"
139
                 type="number"
142
-                step="any"
143
-                name="minPrice"
140
+                setp="any"
144
                 v-model="propertySearch.minPrice"
141
                 v-model="propertySearch.minPrice"
145
               />
142
               />
146
               <div class="input-group-append" @click="clearFilter('minPrice')">
143
               <div class="input-group-append" @click="clearFilter('minPrice')">
161
               <input
158
               <input
162
                 class="form-control"
159
                 class="form-control"
163
                 type="number"
160
                 type="number"
164
-                step="any"
165
-                name="maxPrice"
161
+                setp="any"
166
                 v-model="propertySearch.maxPrice"
162
                 v-model="propertySearch.maxPrice"
167
               />
163
               />
168
               <div class="input-group-append" @click="clearFilter('maxPrice')">
164
               <div class="input-group-append" @click="clearFilter('maxPrice')">
216
       this.clearFilter('propertyType');
212
       this.clearFilter('propertyType');
217
       this.comType = 'All';
213
       this.comType = 'All';
218
     },
214
     },
219
-    salesTypeSelected() {
220
-      this.$emit('updateSearch', this.propertySearch);
221
-    },
222
     PropertyTypeSelected(item) {
215
     PropertyTypeSelected(item) {
223
       this.propertySearch.propertyType = item.target.value;
216
       this.propertySearch.propertyType = item.target.value;
224
-      this.$emit('updateSearch', this.propertySearch);
225
     },
217
     },
226
     ProvinceSelected(item) {
218
     ProvinceSelected(item) {
227
       if (item.target.value !== 'All') {
219
       if (item.target.value !== 'All') {
229
           Object.assign({}, { province: this.propertySearch.province }),
221
           Object.assign({}, { province: this.propertySearch.province }),
230
         );
222
         );
231
       }
223
       }
232
-      this.$emit('updateSearch', this.propertySearch);
233
     },
224
     },
234
     CitySelected(item) {
225
     CitySelected(item) {
235
       if (item.target.value !== 'All') {
226
       if (item.target.value !== 'All') {
243
           ),
234
           ),
244
         );
235
         );
245
       }
236
       }
246
-      this.$emit('updateSearch', this.propertySearch);
247
-    },
248
-    SuburbSeleted() {
249
-      this.$emit('updateSearch', this.propertySearch);
250
     },
237
     },
251
   },
238
   },
252
 };
239
 };

+ 127
- 147
src/components/property/propertySearchPage.vue Parādīt failu

2
   <!-- eslint-disable max-len -->
2
   <!-- eslint-disable max-len -->
3
   <div class="container">
3
   <div class="container">
4
     <br />
4
     <br />
5
-    <div class="row">
6
-      <div class="col-md-2 offset-4">
7
-        <button type="button" @click="SetType('Residential')" class="btn btn-b-n">Residential</button>
8
-      </div>
9
-      <div class="col-md-2">
10
-        <button type="button" @click="SetType('Commercial')" class="btn btn-b-n">Commercial</button>
11
-      </div>
12
-    </div>
13
-    <div class="container col-md-12" v-if="propertySearch.propertyUsageType === 'Residential'">
5
+    <div class="container col-md-12">
14
       <div class="col-sm-12">
6
       <div class="col-sm-12">
15
         <div class="about-img-box">
7
         <div class="about-img-box">
16
           <img
8
           <img
9
+            v-if="propertyUsageType === 'Residential'"
17
             src="img/Pretoria-South-Africa.jpg"
10
             src="img/Pretoria-South-Africa.jpg"
18
             alt="Property Listing"
11
             alt="Property Listing"
19
             class="img-fluid"
12
             class="img-fluid"
20
             style="width:800px;height:400px; border-radius:10px"
13
             style="width:800px;height:400px; border-radius:10px"
21
           />
14
           />
15
+          <img
16
+            v-else
17
+            src="img/Johannesburg-south-africa-1.jpg"
18
+            alt="Property Listing"
19
+            class="img-fluid"
20
+            style="width:800px;height:400px; border-radius:10px"
21
+          />
22
         </div>
22
         </div>
23
         <div class="sinse-box" style="opacity:0.7; border: white solid 3px; border-radius: 15px">
23
         <div class="sinse-box" style="opacity:0.7; border: white solid 3px; border-radius: 15px">
24
           <h3 class="sinse-title">Property Listing</h3>
24
           <h3 class="sinse-title">Property Listing</h3>
25
         </div>
25
         </div>
26
       </div>
26
       </div>
27
       <br />
27
       <br />
28
+      <div class="row">
29
+        <div class="col-md-2 offset-4">
30
+          <button type="button" @click="SetType('Residential')" class="btn btn-b-n">
31
+            Residential
32
+          </button>
33
+        </div>
34
+        <div class="col-md-2">
35
+          <button type="button" @click="SetType('Commercial')" class="btn btn-b-n">
36
+            Commercial
37
+          </button>
38
+        </div>
39
+      </div>
40
+      <br />
41
+      <div class="col-md-10 offset-1">
42
+        <ul class="nav nav-pills-a nav-pills mb-3 section-t3" id="pills-tab" role="tablist">
43
+          <li class="nav-item">
44
+            <a
45
+              class="nav-link active"
46
+              id="pills-video-tab"
47
+              data-toggle="pill"
48
+              href="#pills-video"
49
+              role="tab"
50
+              aria-controls="pills-video"
51
+              aria-selected="true"
52
+              v-on:click="SetSalesType('Sale')"
53
+              >For Sale</a
54
+            >
55
+          </li>
56
+          <li class="nav-item">
57
+            <a
58
+              class="nav-link"
59
+              id="pills-plans-tab"
60
+              data-toggle="pill"
61
+              href="#pills-plans"
62
+              role="tab"
63
+              aria-controls="pills-plans"
64
+              aria-selected="false"
65
+              v-on:click="SetSalesType('Rent')"
66
+              >To Rent</a
67
+            >
68
+          </li>
69
+        </ul>
70
+      </div>
71
+      <br />
72
+      <div class="row">
73
+        <autoComplete class="col-md-11 offset-1" :items="suburbList" @selection="SelectedFilter" />
74
+      </div>
75
+      <br />
76
+      <div class="row">
77
+        <div class="col-md-10 offset-1">
78
+          <button
79
+            type="button"
80
+            class="input-group-text fa fa-search"
81
+            style="color: #60CBEB"
82
+            @click="SearchClick"
83
+          >
84
+            Search
85
+          </button>
86
+        </div>
87
+      </div>
28
       <div class="row">
88
       <div class="row">
29
         <div class="col-md-12">
89
         <div class="col-md-12">
30
-          <h1 class="my-4">About Residential Properties</h1>
90
+          <h1 class="my-4" v-if="propertyUsageType === 'Residential'">
91
+            About Residential Properties
92
+          </h1>
93
+          <h1 class="my-4" v-else>About Commercial Properties</h1>
31
         </div>
94
         </div>
32
-        <div class="container col-md-6 text-left">
95
+        <div class="container col-md-6 text-left" v-if="propertyUsageType === 'Residential'">
33
           <p>
96
           <p>
34
-            Uni-Vate Properties understands the necessity in property-seekers
35
-            to find that perfect fit;
36
-            a home that meets, and exceeds the individual's needs. That is why,
37
-            our dedicated team sources
38
-            a range of property types, whether that home is a rental option near
39
-            a University for a student
40
-            just starting out, or a family home fit for entertaining.
97
+            Uni-Vate Properties understands the necessity in property-seekers to find that perfect
98
+            fit; a home that meets, and exceeds the individual's needs. That is why, our dedicated
99
+            team sources a range of property types, whether that home is a rental option near a
100
+            University for a student just starting out, or a family home fit for entertaining.
41
           </p>
101
           </p>
42
           <p>
102
           <p>
43
-            Looking to sell your residential property, instead? Uni-Vate
44
-            Properties prides itself on
45
-            professionalism and the right expertise, to help you secure the
46
-            best possible deal for your home.
103
+            Looking to sell your residential property, instead? Uni-Vate Properties prides itself on
104
+            professionalism and the right expertise, to help you secure the best possible deal for
105
+            your home.
47
           </p>
106
           </p>
48
           <p>
107
           <p>
49
             Wish to RENT your property?
108
             Wish to RENT your property?
54
             <router-link to="/property/new/Residential/Sale">Click Here</router-link>
113
             <router-link to="/property/new/Residential/Sale">Click Here</router-link>
55
           </p>
114
           </p>
56
         </div>
115
         </div>
57
-        <div class="col-md-4">
116
+        <div class="container col-md-6 text-left" v-else>
58
           <p>
117
           <p>
59
-            <img
60
-              class="img-fluid"
61
-              src="./../../../public/img/residential.jpg"
62
-              alt="About Resale"
63
-              style="border-radius:10px"
64
-            />
65
-          </p>
66
-        </div>
67
-      </div>
68
-      <div>
69
-        <propertyCard v-if="properties.length > 0" name="propertyholder" :properties="properties" />
70
-        <div v-if="properties.length === 0">
71
-          <img src="../../../public/img/no-homes.png" />
72
-          <br />
73
-          <br />
74
-          <p>Sorry no listing where found matching your search</p>
75
-        </div>
76
-      </div>
77
-      <br />
78
-    </div>
79
-
80
-    <div class="container col-md-12" v-if="propertySearch.propertyUsageType === 'Commercial'">
81
-      <div class="col-sm-12">
82
-        <div class="about-img-box">
83
-          <img
84
-            src="img/Johannesburg-south-africa-1.jpg"
85
-            alt="Property Listing"
86
-            class="img-fluid"
87
-            style="width:800px;height:400px; border-radius:10px"
88
-          />
89
-        </div>
90
-        <div class="sinse-box" style="opacity:0.7; border: white solid 3px; border-radius: 15px">
91
-          <h3 class="sinse-title">Property Listing</h3>
92
-        </div>
93
-      </div>
94
-      <br />
95
-      <div class="row">
96
-        <div class="col-md-12">
97
-          <h1 class="my-4">About Commercial Properties</h1>
98
-        </div>
99
-        <div class="container col-md-6 text-left">
100
-          <p>
101
-            Commercial properties are characteristically any larger properties that
102
-            generate profit through leasing or rental activities. These properties
103
-            are typically used to conduct business and provide companies with cut-and-dry
104
-            leasing agreements, which keep their involvement in the maintenance of the property
105
-            to a minimum, so they may focus on the growth of their company.
118
+            Commercial properties are characteristically any larger properties that generate profit
119
+            through leasing or rental activities. These properties are typically used to conduct
120
+            business and provide companies with cut-and-dry leasing agreements, which keep their
121
+            involvement in the maintenance of the property to a minimum, so they may focus on the
122
+            growth of their company.
106
           </p>
123
           </p>
107
           <p>
124
           <p>
108
             Uni-Vate Properties seeks out professional, clean spaces for such companies and acts as
125
             Uni-Vate Properties seeks out professional, clean spaces for such companies and acts as
109
             mediator between the landlord/-lady and the tenants. Uni-Vate Properties provides
126
             mediator between the landlord/-lady and the tenants. Uni-Vate Properties provides
110
-            value-adding service to clients and conducts business with a high standard
111
-            and integrity.
127
+            value-adding service to clients and conducts business with a high standard and
128
+            integrity.
112
           </p>
129
           </p>
113
           <p>
130
           <p>
114
             Wish to RENT your property?
131
             Wish to RENT your property?
122
         <div class="col-md-4">
139
         <div class="col-md-4">
123
           <p>
140
           <p>
124
             <img
141
             <img
142
+              v-if="propertyUsageType === 'Residential'"
143
+              class="img-fluid"
144
+              src="./../../../public/img/residential.jpg"
145
+              alt="About Resale"
146
+              style="border-radius:10px"
147
+            />
148
+            <img
149
+              v-else
125
               class="img-fluid"
150
               class="img-fluid"
126
               src="./../../../public/img/commercial.jpg"
151
               src="./../../../public/img/commercial.jpg"
127
               alt="About Resale"
152
               alt="About Resale"
130
           </p>
155
           </p>
131
         </div>
156
         </div>
132
       </div>
157
       </div>
133
-      <div>
134
-        <propertyCard v-if="properties.length > 0" name="propertyholder" :properties="properties" />
135
-        <div v-if="properties.length === 0">
136
-          <img src="../../../public/img/no-homes.png" />
137
-          <br />
138
-          <br />
139
-          <p>Sorry no listing where found matching your search</p>
140
-        </div>
141
-      </div>
142
       <br />
158
       <br />
143
     </div>
159
     </div>
144
   </div>
160
   </div>
145
 </template>
161
 </template>
146
 <script>
162
 <script>
147
 import { mapState, mapActions } from 'vuex';
163
 import { mapState, mapActions } from 'vuex';
148
-import propertyCard from './propertyCard.vue';
164
+import autoComplete from '../shared/autoComplete.vue';
149
 
165
 
150
 export default {
166
 export default {
151
   name: 'propertysearch',
167
   name: 'propertysearch',
152
   components: {
168
   components: {
153
-    propertyCard,
169
+    autoComplete,
154
   },
170
   },
155
   data() {
171
   data() {
156
     return {
172
     return {
157
-      propertySearch: {
158
-        userName: '',
159
-        salesType: 'All',
160
-        propertyUsageType: 'All',
161
-        propertyType: 'All',
162
-        province: 'All',
163
-        city: 'All',
164
-        suburb: 'All',
165
-        minPrice: 0,
166
-        maxPrice: 0,
167
-      },
173
+      propertyUsageType: 'Residential',
174
+      salesType: 'Sale',
175
+      searchText: '',
168
     };
176
     };
169
   },
177
   },
170
-  mounted() {
171
-    if (typeof this.propertySearch.propertyUsageType === 'undefined') {
172
-      this.propertySearch.propertyUsageType = 'Residential';
173
-    }
178
+  computed: {
179
+    ...mapState('propertySearch', ['suburbs', 'suburbList', 'propertySearch']),
174
   },
180
   },
175
   methods: {
181
   methods: {
176
-    ...mapActions('propertySearch', ['searchProperties']),
182
+    ...mapActions('propertySearch', ['getSuburbs', 'applyFilter']),
177
     SetType(item) {
183
     SetType(item) {
178
-      this.propertySearch.propertyUsageType = item;
184
+      this.propertyUsageType = item;
179
     },
185
     },
180
-  },
181
-  computed: {
182
-    ...mapState('propertySearch', ['properties']),
183
-    ...mapState('authentication', ['username']),
184
-    ParamsChanged() {
185
-      if (Object.keys(this.$route.query).length === 0) {
186
-        if (this.propertySearch.propertyUsageType === 'All') {
187
-          // eslint-disable-next-line vue/no-side-effects-in-computed-properties
188
-          this.propertySearch.propertyUsageType = 'Residential';
189
-          // eslint-disable-next-line vue/no-side-effects-in-computed-properties
190
-          this.propertySearch.keyword = 'All';
191
-          // eslint-disable-next-line vue/no-side-effects-in-computed-properties
192
-          this.propertySearch.salesType = 'All';
193
-        }
194
-      }
195
-      if (Object.keys(this.$route.query).length > 0) {
196
-        if (Object.keys(this.$route.query).length > 2) {
197
-          // eslint-disable-next-line vue/no-side-effects-in-computed-properties
198
-          this.propertySearch = this.$route.query;
199
-        } else {
200
-          if (this.$route.query.salesType) {
201
-            // eslint-disable-next-line vue/no-side-effects-in-computed-properties
202
-            this.propertySearch.salesType = this.$route.query.salesType;
203
-          }
204
-          if (this.$route.query.propertyUsageType) {
205
-            // eslint-disable-next-line vue/no-side-effects-in-computed-properties
206
-            this.propertySearch.propertyUsageType = this.$route.query.propertyUsageType;
207
-          }
208
-        }
209
-      }
210
-      if (this.propertySearch.keyword === '') {
211
-        // eslint-disable-next-line vue/no-side-effects-in-computed-properties
212
-        this.propertySearch.keyword = 'All';
213
-      }
214
-      if (this.username) {
215
-        // eslint-disable-next-line vue/no-side-effects-in-computed-properties
216
-        this.propertySearch.userName = this.username;
217
-      } else {
218
-        // eslint-disable-next-line vue/no-side-effects-in-computed-properties
219
-        this.propertySearch.userName = 'Unknown';
220
-      }
221
-      this.searchProperties(this.propertySearch);
222
-      return null;
186
+    SearchClick() {
187
+      const item = this.suburbs.find(s => s.display === this.searchText);
188
+      console.log(JSON.stringify(item));
189
+      this.propertySearch.province = item.province;
190
+      this.propertySearch.city = item.city;
191
+      this.propertySearch.suburb = item.suburb;
192
+      this.propertySearch.propertyUsageType = this.propertyUsageType;
193
+      this.propertySearch.salesType = this.salesType;
194
+
195
+      this.$router.push('/property/propertySearch/results');
223
     },
196
     },
224
-  },
225
-  watch: {
226
-    ParamsChanged() {
227
-      return null;
197
+    Filter() {
198
+      this.applyFilter(this.searchText);
228
     },
199
     },
200
+    SetSalesType(value) {
201
+      this.salesType = value;
202
+    },
203
+    SelectedFilter(item) {
204
+      this.searchText = item;
205
+    },
206
+  },
207
+  mounted() {
208
+    this.getSuburbs();
229
   },
209
   },
230
 };
210
 };
231
 </script>
211
 </script>

+ 83
- 0
src/components/property/propertySearchResults.vue Parādīt failu

1
+<template>
2
+  <!-- eslint-disable max-len -->
3
+  <div class="container">
4
+    <br />
5
+    <div class="container col-md-12">
6
+      <div class="col-sm-12">
7
+        <div class="about-img-box">
8
+          <img
9
+            v-if="propertySearch.propertyUsageType === 'Residential'"
10
+            src="img/Pretoria-South-Africa.jpg"
11
+            alt="Property Listing"
12
+            class="img-fluid"
13
+            style="width:800px;height:400px; border-radius:10px"
14
+          />
15
+          <img
16
+            v-else
17
+            src="img/Johannesburg-south-africa-1.jpg"
18
+            alt="Property Listing"
19
+            class="img-fluid"
20
+            style="width:800px;height:400px; border-radius:10px"
21
+          />
22
+        </div>
23
+        <div class="sinse-box" style="opacity:0.7; border: white solid 3px; border-radius: 15px">
24
+          <h3 class="sinse-title">Property Listing</h3>
25
+        </div>
26
+      </div>
27
+      <br />
28
+      <div>
29
+        <propertyCard v-if="properties.length > 0" name="propertyholder" :properties="properties" />
30
+        <div v-if="properties.length === 0">
31
+          <img src="../../../public/img/no-homes.png" />
32
+          <br />
33
+          <br />
34
+          <p>Sorry no listing where found matching your search</p>
35
+        </div>
36
+      </div>
37
+      <br />
38
+    </div>
39
+  </div>
40
+</template>
41
+<script>
42
+import { mapState, mapActions } from 'vuex';
43
+import propertyCard from './propertyCard.vue';
44
+
45
+export default {
46
+  name: 'propertysearch',
47
+  components: {
48
+    propertyCard,
49
+  },
50
+  data() {
51
+    return {};
52
+  },
53
+  mounted() {
54
+    if (typeof this.propertySearch.propertyUsageType === 'undefined') {
55
+      this.propertySearch.propertyUsageType = 'Residential';
56
+    }
57
+    this.searchProperties(this.propertySearch);
58
+  },
59
+  methods: {
60
+    ...mapActions('propertySearch', ['searchProperties']),
61
+    SetType(item) {
62
+      this.propertySearch.propertyUsageType = item;
63
+    },
64
+  },
65
+  computed: {
66
+    ...mapState('propertySearch', ['properties', 'propertySearch']),
67
+    ...mapState('authentication', ['username']),
68
+    ParamsChanged() {
69
+      if (typeof this.propertySearch.propertyUsageType === 'undefined') {
70
+        // eslint-disable-next-line vue/no-side-effects-in-computed-properties
71
+        this.propertySearch.propertyUsageType = 'Residential';
72
+      }
73
+      this.searchProperties(this.propertySearch);
74
+      return null;
75
+    },
76
+  },
77
+  watch: {
78
+    ParamsChanged() {
79
+      return null;
80
+    },
81
+  },
82
+};
83
+</script>

+ 168
- 199
src/components/property/propertyeditPage.vue Parādīt failu

35
       <div class="row mb-3">
35
       <div class="row mb-3">
36
         <div class="container col-md-10" style="text-align:left">
36
         <div class="container col-md-10" style="text-align:left">
37
           <form id="mainForm">
37
           <form id="mainForm">
38
+            <div class="form-goup row">
39
+              <div class="col-md-4">
40
+                <field :type="'datetime'" :mayEdit="true" />
41
+              </div>
42
+            </div>
38
             <div class="form-goup row">
43
             <div class="form-goup row">
39
               <div class="col-md-4">
44
               <div class="col-md-4">
40
                 <label>Usage Type</label>
45
                 <label>Usage Type</label>
41
                 <div class="input-group-prepend">
46
                 <div class="input-group-prepend">
42
-                  <!-- <span class="input-group-text" style="color: #60CBEB">
43
-                    <b>U</b>
44
-                  </span>-->
45
                   <input
47
                   <input
46
                     class="form-control"
48
                     class="form-control"
47
                     type="text"
49
                     type="text"
53
                 </div>
55
                 </div>
54
               </div>
56
               </div>
55
             </div>
57
             </div>
58
+            <br />
56
             <div class="form-group row">
59
             <div class="form-group row">
57
               <div class="col-md-4">
60
               <div class="col-md-4">
58
                 <label>Sale Type</label>
61
                 <label>Sale Type</label>
64
                   :mayEdit="mayEdit"
67
                   :mayEdit="mayEdit"
65
                   @UpdateValue="UpdateValue"
68
                   @UpdateValue="UpdateValue"
66
                 />
69
                 />
70
+                <div v-if="showPropertyTypeError">
71
+                  <p class="alert myError">Please select a Sales Type</p>
72
+                </div>
67
               </div>
73
               </div>
68
             </div>
74
             </div>
69
             <div class="form-group row">
75
             <div class="form-group row">
70
               <div class="col-md-4">
76
               <div class="col-md-4">
71
-                <label>Property Type</label>
72
-                <propField
73
-                  :display="property.propertyType ? property.propertyType.description : ''"
74
-                  :editType="'selector'"
75
-                  :arrayObject="propertyTypes"
76
-                  :propertyName="'propertyType'"
77
+                <label>Property Type *</label>
78
+                <field
79
+                  :type="'select'"
80
+                  :selectOptions="propertyTypes"
81
+                  :selectValue="'id'"
82
+                  :selectText="'description'"
83
+                  v-model="property.propertyType"
77
                   :mayEdit="mayEdit"
84
                   :mayEdit="mayEdit"
78
-                  @UpdateValue="UpdateValue"
79
                 />
85
                 />
80
               </div>
86
               </div>
81
               <div v-if="propertyType === 'Commercial'" class="col-md-4">
87
               <div v-if="propertyType === 'Commercial'" class="col-md-4">
82
                 <label>Property Name</label>
88
                 <label>Property Name</label>
83
-                <propField
84
-                  :display="property.propertyName"
85
-                  :editType="'text'"
86
-                  :propertyName="'propertyName'"
87
-                  :mayEdit="mayEdit"
88
-                  @UpdateValue="UpdateValue"
89
-                />
89
+                <field :type="'text'" v-model="property.propertyName" :mayEdit="mayEdit" />
90
               </div>
90
               </div>
91
               <div v-if="propertyType === 'Commercial'" class="col-md-4">
91
               <div v-if="propertyType === 'Commercial'" class="col-md-4">
92
                 <label>Unit</label>
92
                 <label>Unit</label>
93
-                <propField
94
-                  :label="'Property Type'"
95
-                  :display="property.unit"
96
-                  :editType="'text'"
97
-                  :propertyName="'unit'"
98
-                  :mayEdit="mayEdit"
99
-                  @UpdateValue="UpdateValue"
100
-                />
93
+                <field :type="'text'" v-model="property.unit" :mayEdit="mayEdit" />
101
               </div>
94
               </div>
102
             </div>
95
             </div>
103
             <div class="form-group row">
96
             <div class="form-group row">
104
               <div class="col-md-6" style="margin-bottom: 1em">
97
               <div class="col-md-6" style="margin-bottom: 1em">
105
                 <label>Street Number</label>
98
                 <label>Street Number</label>
106
-                <propField
107
-                  :display="property.addressLine1"
108
-                  :editType="'text'"
109
-                  :propertyName="'addressLine1'"
110
-                  :mayEdit="mayEdit"
111
-                  @UpdateValue="UpdateValue"
112
-                />
99
+                <field :type="'text'" v-model="property.addressLine1" :mayEdit="mayEdit" />
113
               </div>
100
               </div>
114
               <div class="col-md-6" style="margin-bottom: 1em">
101
               <div class="col-md-6" style="margin-bottom: 1em">
115
                 <label>Street Name</label>
102
                 <label>Street Name</label>
116
-                <propField
117
-                  :display="property.addressLine2"
118
-                  :editType="'text'"
119
-                  :propertyName="'addressLine2'"
120
-                  :mayEdit="mayEdit"
121
-                  @UpdateValue="UpdateValue"
122
-                />
103
+                <field :type="'text'" v-model="property.addressLine2" :mayEdit="mayEdit" />
123
               </div>
104
               </div>
124
               <div class="col-md-6" style="margin-bottom: 1em">
105
               <div class="col-md-6" style="margin-bottom: 1em">
125
-                <label>Province</label>
126
-                <propField
106
+                <label>Province *</label>
107
+                <!-- <propField
127
                   :display="property.province ? property.province.description : ''"
108
                   :display="property.province ? property.province.description : ''"
128
                   :editType="'selector'"
109
                   :editType="'selector'"
129
                   :arrayObject="provinces"
110
                   :arrayObject="provinces"
130
                   :propertyName="'province'"
111
                   :propertyName="'province'"
131
                   :mayEdit="mayEdit"
112
                   :mayEdit="mayEdit"
132
                   @UpdateValue="UpdateValue"
113
                   @UpdateValue="UpdateValue"
114
+                />-->
115
+                <field
116
+                  :type="'select'"
117
+                  :selectOptions="provinces"
118
+                  :selectValue="'id'"
119
+                  :selectText="'description'"
120
+                  v-model="property.province"
121
+                  :mayEdit="mayEdit"
122
+                  @change="provChanged"
133
                 />
123
                 />
124
+                <div v-if="showProvinceError">
125
+                  <p class="alert myError">Please select a Province</p>
126
+                </div>
134
               </div>
127
               </div>
135
-
136
               <div class="col-md-6" style="margin-bottom: 1em">
128
               <div class="col-md-6" style="margin-bottom: 1em">
137
-                <label>City</label>
138
-                <propField
129
+                <label>City *</label>
130
+                <!-- <propField
139
                   :display="property.city ? property.city.description : ''"
131
                   :display="property.city ? property.city.description : ''"
140
                   :editType="'selector'"
132
                   :editType="'selector'"
141
                   :arrayObject="cities"
133
                   :arrayObject="cities"
142
                   :propertyName="'city'"
134
                   :propertyName="'city'"
143
                   :mayEdit="mayEdit"
135
                   :mayEdit="mayEdit"
144
                   @UpdateValue="UpdateValue"
136
                   @UpdateValue="UpdateValue"
137
+                />-->
138
+                <field
139
+                  :type="'select'"
140
+                  :selectOptions="cities"
141
+                  :selectValue="'id'"
142
+                  :selectText="'description'"
143
+                  v-model="property.city"
144
+                  :mayEdit="mayEdit"
145
+                  @change="cityChanged"
145
                 />
146
                 />
147
+                <div v-if="showCityError">
148
+                  <p class="alert myError">Please select a City</p>
149
+                </div>
146
               </div>
150
               </div>
147
               <div class="col-md-6" style="margin-bottom: 1em">
151
               <div class="col-md-6" style="margin-bottom: 1em">
148
-                <label>Suburb</label>
149
-                <propField
152
+                <label>Suburb *</label>
153
+                <!-- <propField
150
                   :display="property.suburb ? property.suburb.description : ''"
154
                   :display="property.suburb ? property.suburb.description : ''"
151
                   :editType="'selector'"
155
                   :editType="'selector'"
152
                   :arrayObject="suburbs"
156
                   :arrayObject="suburbs"
153
                   :propertyName="'suburb'"
157
                   :propertyName="'suburb'"
154
                   :mayEdit="mayEdit"
158
                   :mayEdit="mayEdit"
155
                   @UpdateValue="UpdateValue"
159
                   @UpdateValue="UpdateValue"
160
+                />-->
161
+                <field
162
+                  :type="'select'"
163
+                  :selectOptions="suburbs"
164
+                  :selectValue="'id'"
165
+                  :selectText="'description'"
166
+                  v-model="property.suburb"
167
+                  :mayEdit="mayEdit"
168
+                  @change="suburbChanged"
156
                 />
169
                 />
170
+                <div v-if="showSuburbError">
171
+                  <p class="alert myError">Please select a suburb</p>
172
+                </div>
157
               </div>
173
               </div>
158
               <div class="col-md-6" style="margin-bottom: 1em">
174
               <div class="col-md-6" style="margin-bottom: 1em">
159
                 <label>Postal Code</label>
175
                 <label>Postal Code</label>
160
-                <propField
161
-                  :display="property.addressLine3"
162
-                  :editType="'text'"
163
-                  :propertyName="'addressLine3'"
164
-                  :mayEdit="mayEdit"
165
-                  @UpdateValue="UpdateValue"
166
-                />
176
+                <field :type="'text'" v-model="property.addressLine3" :mayEdit="mayEdit" />
167
               </div>
177
               </div>
168
             </div>
178
             </div>
169
 
179
 
171
               <div class="col-md-6">
181
               <div class="col-md-6">
172
                 <label v-if="salesType === 'Rental'">Rental Price</label>
182
                 <label v-if="salesType === 'Rental'">Rental Price</label>
173
                 <label v-if="salesType !== 'Rental'">Sales Price</label>
183
                 <label v-if="salesType !== 'Rental'">Sales Price</label>
174
-                <propField
184
+                <field :type="'number'" v-model="property.price" :mayEdit="mayEdit" />
185
+                <!-- <propField
175
                   :display="String(property.price)"
186
                   :display="String(property.price)"
176
                   :editType="'number'"
187
                   :editType="'number'"
177
                   :propertyName="'price'"
188
                   :propertyName="'price'"
178
                   :mayEdit="mayEdit"
189
                   :mayEdit="mayEdit"
179
                   @UpdateValue="UpdateValue"
190
                   @UpdateValue="UpdateValue"
180
-                />
191
+                />-->
181
               </div>
192
               </div>
182
               <div class="col-md-6" v-if="salesType === 'Rental'">
193
               <div class="col-md-6" v-if="salesType === 'Rental'">
183
                 <label>Per</label>
194
                 <label>Per</label>
184
-                <propField
195
+                <!-- <propField
185
                   :display="property.pricePer"
196
                   :display="property.pricePer"
186
                   :editType="'selector'"
197
                   :editType="'selector'"
187
                   :propertyName="'pricePer'"
198
                   :propertyName="'pricePer'"
188
                   :arrayObject="pricePerArr"
199
                   :arrayObject="pricePerArr"
189
                   :mayEdit="mayEdit"
200
                   :mayEdit="mayEdit"
190
                   @UpdateValue="UpdateValue"
201
                   @UpdateValue="UpdateValue"
202
+                />-->
203
+                <field
204
+                  :type="'selectlist'"
205
+                  :selectOptions="pricePerArr"
206
+                  v-model="property.pricePer"
207
+                  :mayEdit="mayEdit"
191
                 />
208
                 />
192
               </div>
209
               </div>
193
             </div>
210
             </div>
196
                 <label for="Property Description">Description</label>
213
                 <label for="Property Description">Description</label>
197
                 <vue-editor v-model="property.description" :editor-toolbar="customToolbar" />
214
                 <vue-editor v-model="property.description" :editor-toolbar="customToolbar" />
198
                 <br />
215
                 <br />
199
-                <p>* A listing fee of R380 including VAT is payable to list your Property on the Uni-Vate website</p>
216
+                <p>
217
+                  * A listing fee of R380 including VAT is payable to list your Property on the
218
+                  Uni-Vate website
219
+                </p>
200
               </div>
220
               </div>
201
             </div>
221
             </div>
202
             <div class="form-group row" />
222
             <div class="form-group row" />
203
             <div
223
             <div
204
               class="col-md-6"
224
               class="col-md-6"
205
-              v-if="propertyType === 'Residential' & propertyOverviewFields.length > 0"
225
+              v-if="(propertyType === 'Residential') & (propertyOverviewFields.length > 0)"
206
             >
226
             >
207
               <div v-for="(item, i) in propertyOverviewFields[0].fields" :key="item.id">
227
               <div v-for="(item, i) in propertyOverviewFields[0].fields" :key="item.id">
208
                 <label>{{ item.name }}</label>
228
                 <label>{{ item.name }}</label>
243
                   @UpdateValue="UpdateValue"
263
                   @UpdateValue="UpdateValue"
244
                 />
264
                 />
245
               </div>
265
               </div>
266
+              <br />
246
             </div>
267
             </div>
247
             <div class="form-group row" />
268
             <div class="form-group row" />
248
             <div class="row">
269
             <div class="row">
255
             <div class="form-group row">
276
             <div class="form-group row">
256
               <div class="col-md-12">
277
               <div class="col-md-12">
257
                 <label>Virtual Tour (URL)</label>
278
                 <label>Virtual Tour (URL)</label>
258
-                <propField
259
-                  :display="property.virtualTour"
260
-                  :editType="'text'"
261
-                  :propertyName="'virtualTour'"
262
-                  :mayEdit="mayEdit"
263
-                  @UpdateValue="UpdateValue"
264
-                />
279
+                <field :type="'text'" v-model="property.virtualTour" :mayEdit="mayEdit" />
265
               </div>
280
               </div>
266
             </div>
281
             </div>
267
             <div class="form-group row">
282
             <div class="form-group row">
268
               <div class="col-md-12">
283
               <div class="col-md-12">
269
                 <label>Video (URL)</label>
284
                 <label>Video (URL)</label>
270
-                <propField
271
-                  :display="property.video"
272
-                  :editType="'text'"
273
-                  :propertyName="'video'"
274
-                  :mayEdit="mayEdit"
275
-                  @UpdateValue="UpdateValue"
276
-                />
285
+                <field :type="'text'" v-model="property.video" :mayEdit="mayEdit" />
277
               </div>
286
               </div>
278
             </div>
287
             </div>
279
             <div class="form-group row">
288
             <div class="form-group row">
300
                 />
309
                 />
301
                 <span
310
                 <span
302
                   v-if="!img.isDeleted && mayEdit"
311
                   v-if="!img.isDeleted && mayEdit"
303
-                  class="input-group-text"
312
+                  class="input-group-text spanCursor"
304
                   align="center"
313
                   align="center"
305
                   @click="DeleteImage(img)"
314
                   @click="DeleteImage(img)"
315
+                  style="width:150px"
306
                 >
316
                 >
307
                   <eva-icon name="trash-2-outline" fill="#60CBEB"></eva-icon>Delete
317
                   <eva-icon name="trash-2-outline" fill="#60CBEB"></eva-icon>Delete
308
                 </span>
318
                 </span>
321
               class="btn btn-b-n"
331
               class="btn btn-b-n"
322
               style="width: 85px; height:40px;"
332
               style="width: 85px; height:40px;"
323
               :disabled="!mayEdit"
333
               :disabled="!mayEdit"
324
-            >Save</button>
334
+            >
335
+              Save
336
+            </button>
325
             <button
337
             <button
326
               type="button"
338
               type="button"
327
               @click="Close()"
339
               @click="Close()"
328
               class="btn btn-b-n"
340
               class="btn btn-b-n"
329
               style="width: 85px; height:40px;"
341
               style="width: 85px; height:40px;"
330
-            >Close</button>
342
+              :disabled="
343
+                showPropertyTypeError || showProvinceError || showCityError || showSuburbError
344
+              "
345
+            >
346
+              Close
347
+            </button>
348
+            <div
349
+              v-if="showPropertyTypeError || showProvinceError || showCityError || showSuburbError"
350
+            >
351
+              <p class="alert myError">
352
+                Missing fields. Please fill in all required fields. Marked with *
353
+              </p>
354
+            </div>
331
             <div v-if="wait" id="preloader"></div>
355
             <div v-if="wait" id="preloader"></div>
332
           </form>
356
           </form>
333
         </div>
357
         </div>
339
 <script>
363
 <script>
340
 import { mapState, mapActions } from 'vuex';
364
 import { mapState, mapActions } from 'vuex';
341
 import { VueEditor } from 'vue2-editor';
365
 import { VueEditor } from 'vue2-editor';
342
-import { setTimeout } from 'timers';
343
 import ImageLoad from './propertyImage.vue';
366
 import ImageLoad from './propertyImage.vue';
344
 import propField from './propertyFieldEditor.vue';
367
 import propField from './propertyFieldEditor.vue';
368
+import field from '../shared/fieldEditor.vue';
345
 
369
 
346
 export default {
370
 export default {
347
   name: 'PropertyEdit',
371
   name: 'PropertyEdit',
349
     ImageLoad,
373
     ImageLoad,
350
     VueEditor,
374
     VueEditor,
351
     propField,
375
     propField,
376
+    field,
352
   },
377
   },
353
   data() {
378
   data() {
354
     return {
379
     return {
361
       customToolbar: [
386
       customToolbar: [
362
         [{ header: [false, 1, 2, 3, 4, 5, 6] }],
387
         [{ header: [false, 1, 2, 3, 4, 5, 6] }],
363
         ['bold', 'italic', 'underline', 'strike'],
388
         ['bold', 'italic', 'underline', 'strike'],
364
-        [
365
-          { align: '' },
366
-          { align: 'center' },
367
-          { align: 'right' },
368
-          { align: 'justify' },
369
-        ],
389
+        [{ align: '' }, { align: 'center' }, { align: 'right' }, { align: 'justify' }],
370
         [{ list: 'ordered' }, { list: 'bullet' }, { list: 'check' }],
390
         [{ list: 'ordered' }, { list: 'bullet' }, { list: 'check' }],
371
         [{ script: 'sub' }, { script: 'super' }],
391
         [{ script: 'sub' }, { script: 'super' }],
372
         [{ indent: '-1' }, { indent: '+1' }],
392
         [{ indent: '-1' }, { indent: '+1' }],
375
         { id: 1, description: 'Sale' },
395
         { id: 1, description: 'Sale' },
376
         { id: 2, description: 'Rental' },
396
         { id: 2, description: 'Rental' },
377
       ],
397
       ],
378
-      pricePerArr: [
379
-        { id: 'Month', description: 'Month' },
380
-        { id: 'Day', description: 'Day' },
381
-      ],
398
+      pricePerArr: ['Month', 'Day'],
399
+      yesno: ['Yes', 'No'],
400
+      showPropertyTypeError: false,
401
+      showProvinceError: false,
402
+      showCityError: false,
403
+      showSuburbError: false,
382
     };
404
     };
383
   },
405
   },
384
   methods: {
406
   methods: {
389
       'getListsForPropertyEdit',
411
       'getListsForPropertyEdit',
390
     ]),
412
     ]),
391
     ...mapActions('property', ['getPropertyTypes']),
413
     ...mapActions('property', ['getPropertyTypes']),
392
-    ...mapActions('propertyEdit', [
393
-      'getSavedPropertyData',
394
-      'updateProperty',
395
-      'mayEditProperty',
396
-    ]),
414
+    ...mapActions('propertyEdit', ['getSavedPropertyData', 'updateProperty', 'mayEditProperty']),
415
+    provChanged(item) {
416
+      this.getCities(Object.assign({}, { province: item.description }));
417
+      this.property.city = null;
418
+      this.property.suburb = null;
419
+      this.property.addressLine3 = '';
420
+      this.showProvinceError = false;
421
+      this.showCityError = true;
422
+      this.showSuburbError = true;
423
+    },
424
+    cityChanged(item) {
425
+      this.getSuburbs(
426
+        Object.assign(
427
+          {},
428
+          {
429
+            province: this.property.province.description,
430
+            city: item.description,
431
+          },
432
+        ),
433
+      );
434
+      this.property.suburb = null;
435
+      this.property.addressLine3 = '';
436
+      this.showCityError = false;
437
+      this.showSuburbError = true;
438
+    },
439
+    suburbChanged(item) {
440
+      const newSuburb = this.suburbs.find(p => p.description === item.description);
441
+      this.property.addressLine3 = newSuburb.postalCode;
442
+      this.showSuburbError = false;
443
+    },
444
+    salesTypeChanged(item) {
445
+      console.log(item);
446
+    },
397
     UpdateValue(item) {
447
     UpdateValue(item) {
398
       if (item.isUDF) {
448
       if (item.isUDF) {
399
         if (item.isPropOverview) {
449
         if (item.isPropOverview) {
400
-          this.propertyOverviewFields[0].fields[item.arrayIndex].value =            item.value;
450
+          this.propertyOverviewFields[0].fields[item.arrayIndex].value = item.value;
401
         } else if (item.isPropOverview === false) {
451
         } else if (item.isPropOverview === false) {
402
-          this.propertyFields[item.arrayIndex].fields[
403
-            item.arrayItemIndex
404
-          ].value = item.value;
452
+          this.propertyFields[item.arrayIndex].fields[item.arrayItemIndex].value = item.value;
405
         }
453
         }
406
       } else if (!item.isUDF) {
454
       } else if (!item.isUDF) {
407
         if (item.fieldName) {
455
         if (item.fieldName) {
412
             } else isSaleValue = false;
460
             } else isSaleValue = false;
413
             this.property.isSale = isSaleValue;
461
             this.property.isSale = isSaleValue;
414
           }
462
           }
415
-          if (item.fieldName === 'propertyType') {
416
-            this.property.propertyType = this.propertyTypes.find(
417
-              pt => pt.description === item.value,
418
-            );
419
-            this.property.propertyTypeId = this.property.propertyType.id;
420
-          }
421
-          if (item.fieldName === 'propertyName') {
422
-            this.property.propertyName = item.value;
423
-          }
424
-          if (item.fieldName === 'unit') {
425
-            this.property.unit = item.value;
426
-          }
427
-          if (item.fieldName === 'addressLine1') {
428
-            this.property.addressLine1 = item.value;
429
-          }
430
-          if (item.fieldName === 'addressLine2') {
431
-            this.property.addressLine2 = item.value;
432
-          }
433
-          if (item.fieldName === 'addressLine3') {
434
-            this.property.addressLine3 = item.value;
435
-          }
436
-          if (item.fieldName === 'province') {
437
-            if (item.value !== '') {
438
-              this.property.province = this.provinces.find(
439
-                p => p.description === item.value,
440
-              );
441
-              this.property.provinceId = this.property.province.id;
442
-              this.getCities(Object.assign({}, { province: item.value }));
443
-              this.property.city = null;
444
-              this.property.suburb = null;
445
-              this.property.addressLine3 = '';
446
-            } else {
447
-              this.property.province = null;
448
-              this.property.city = null;
449
-              this.property.suburb = null;
450
-              this.property.addressLine3 = '';
451
-              this.cities = [];
452
-              this.suburbs = [];
453
-            }
454
-          }
455
-          if (item.fieldName === 'city') {
456
-            if (item.value !== '') {
457
-              const newCity = this.cities.find(
458
-                p => p.description === item.value,
459
-              );
460
-              this.property.city = newCity;
461
-              this.property.cityId = newCity.id;
462
-              this.getSuburbs(
463
-                Object.assign(
464
-                  {},
465
-                  {
466
-                    province: this.property.province.description,
467
-                    city: item.value,
468
-                  },
469
-                ),
470
-              );
471
-              this.property.suburb = null;
472
-              this.property.addressLine3 = '';
473
-            } else {
474
-              this.property.city = null;
475
-              this.property.suburb = null;
476
-              this.property.addressLine3 = '';
477
-              this.suburbs = [];
478
-            }
479
-          }
480
-          if (item.fieldName === 'suburb') {
481
-            if (item.value !== '') {
482
-              const newSuburb = this.suburbs.find(
483
-                p => p.description === item.value,
484
-              );
485
-              this.property.suburb = newSuburb;
486
-              this.property.suburbId = newSuburb.id;
487
-              this.property.addressLine3 = this.property.suburb.postalCode;
488
-            } else {
489
-              this.property.suburb = null;
490
-              this.property.addressLine3 = '';
491
-            }
492
-          }
493
-          if (item.fieldName === 'price') {
494
-            this.property.price = item.value;
495
-          }
496
-          if (item.fieldName === 'pricePer') {
497
-            this.property.pricePer = item.value;
498
-          }
499
-          if (item.fieldName === 'virtualTour') {
500
-            this.property.virtualTour = item.value;
501
-          }
502
-          if (item.fieldName === 'video') {
503
-            this.property.video = item.value;
504
-          }
505
         }
463
         }
506
       }
464
       }
507
     },
465
     },
509
       this.propertyType = this.property.propertyUsageType;
467
       this.propertyType = this.property.propertyUsageType;
510
     },
468
     },
511
     SubmitData() {
469
     SubmitData() {
512
-      this.wait = true;
470
+      if (this.property.propertyTypeId === 0) {
471
+        this.showPropertyTypeError = true;
472
+      }
473
+      if (this.property.province === null) {
474
+        this.showProvinceError = true;
475
+      }
476
+      if (this.property.city === null) {
477
+        this.showCityError = true;
478
+      }
479
+      if (this.property.suburb === null) {
480
+        this.showSuburbError = true;
481
+      }
513
 
482
 
483
+      if (
484
+        this.showPropertyTypeError
485
+        || this.showProvinceError
486
+        || this.showCityError
487
+        || this.showSuburbError
488
+      ) {
489
+        return;
490
+      }
491
+      this.wait = true;
514
       this.newPropertyImages.propertyId = this.property.id;
492
       this.newPropertyImages.propertyId = this.property.id;
515
       this.newPropertyImages.Images = [];
493
       this.newPropertyImages.Images = [];
516
       // eslint-disable-next-line no-plusplus
494
       // eslint-disable-next-line no-plusplus
524
           isDefault: setAsDefault,
502
           isDefault: setAsDefault,
525
         });
503
         });
526
       }
504
       }
527
-
528
       this.property.propertyImages = this.propertyImages;
505
       this.property.propertyImages = this.propertyImages;
529
       // eslint-disable-next-line no-plusplus
506
       // eslint-disable-next-line no-plusplus
530
       for (let i = 0; i < this.property.propertyImages.length; i++) {
507
       for (let i = 0; i < this.property.propertyImages.length; i++) {
535
       this.updateProperty({
512
       this.updateProperty({
536
         property: this.property,
513
         property: this.property,
537
         images: this.newPropertyImages,
514
         images: this.newPropertyImages,
515
+      }).then(() => {
516
+        this.$router.push('/properties');
538
       });
517
       });
539
-
540
-      // console.log(JSON.stringify(this.newPropertyImages));
541
-
542
-      // Need to change to promis.
543
-      setTimeout(
544
-        () => this.$router.push({
545
-            path: '/property/admin/list/my',
546
-          }),
547
-        5000,
548
-      );
549
     },
518
     },
550
     Close() {
519
     Close() {
551
-      this.$router.push('/property/admin/list/my');
520
+      this.$router.push('/properties');
552
     },
521
     },
553
     loadedImages(values) {
522
     loadedImages(values) {
554
       this.images = values;
523
       this.images = values;
625
 </script>
594
 </script>
626
 
595
 
627
 <style>
596
 <style>
628
-span {
597
+.spanCursor {
629
   cursor: pointer;
598
   cursor: pointer;
630
 }
599
 }
631
 .opacity {
600
 .opacity {

+ 30
- 0
src/components/shared/alertPage.vue Parādīt failu

1
+<template>
2
+  <div>
3
+    <br />
4
+    <Alert :text="message" :type="type" />
5
+    <div class="row">
6
+      <div class="offset-3 col-md-2">
7
+        <button type="button" class="form-control btn btn-primary" @click="goBack()">Back</button>
8
+      </div>
9
+    </div>
10
+  </div>
11
+</template>
12
+
13
+<script>
14
+import { mapState } from 'vuex';
15
+import Alert from './alert.vue';
16
+
17
+export default {
18
+  components: {
19
+    Alert,
20
+  },
21
+  computed: {
22
+    ...mapState('alert', ['message', 'type']),
23
+  },
24
+  methods: {
25
+    goBack() {
26
+      this.$router.go(-1);
27
+    },
28
+  },
29
+};
30
+</script>

+ 81
- 0
src/components/shared/autoComplete.vue Parādīt failu

1
+<template>
2
+  <div class="autocomplete">
3
+    <input type="text" v-model="search" @input="onChange" class="form-control" />
4
+    <ul v-show="isOpen" class="autocomplete-results">
5
+      <li
6
+        v-for="(result, i) in results"
7
+        :key="i"
8
+        class="autocomplete-result"
9
+        @click="setResult(result)"
10
+      >
11
+        {{ result }}
12
+      </li>
13
+    </ul>
14
+  </div>
15
+</template>
16
+
17
+<script>
18
+export default {
19
+  props: {
20
+    items: {
21
+      type: Array,
22
+      required: false,
23
+      default: () => [],
24
+    },
25
+  },
26
+  data() {
27
+    return {
28
+      search: '',
29
+      results: [],
30
+      isOpen: false,
31
+    };
32
+  },
33
+  methods: {
34
+    onChange() {
35
+      if (this.search.length >= 3) {
36
+        this.isOpen = true;
37
+        this.filterResults();
38
+      } else {
39
+        this.isOpen = false;
40
+      }
41
+    },
42
+    filterResults() {
43
+      this.results = this.items.filter(
44
+        item => item.toLowerCase().indexOf(this.search.toLowerCase()) > -1,
45
+      );
46
+    },
47
+    setResult(result) {
48
+      this.search = result;
49
+      this.isOpen = false;
50
+      this.$emit('selection', this.search);
51
+    },
52
+  },
53
+};
54
+</script>
55
+
56
+<style>
57
+.autocomplete {
58
+  position: relative;
59
+  width: 130px;
60
+}
61
+
62
+.autocomplete-results {
63
+  padding: 0;
64
+  margin: 0;
65
+  border: 1px solid #eeeeee;
66
+  height: 120px;
67
+  overflow: auto;
68
+}
69
+
70
+.autocomplete-result {
71
+  list-style: none;
72
+  text-align: left;
73
+  padding: 4px 2px;
74
+  cursor: pointer;
75
+}
76
+
77
+.autocomplete-result:hover {
78
+  background-color: #60cbeb;
79
+  color: white;
80
+}
81
+</style>

+ 118
- 0
src/components/shared/fieldEditor.vue Parādīt failu

1
+<template>
2
+  <div>
3
+    <div v-if="!edit" class="input-group-prepend">
4
+      <input class="form-control" v-model="myDisplay" disabled />
5
+      <span
6
+        v-if="mayEdit"
7
+        @click="EditClick()"
8
+        class="input-group-text spanCursor"
9
+        style="color: #60CBEB"
10
+      >
11
+        <eva-icon name="edit-outline" fill="#60CBEB"></eva-icon>
12
+      </span>
13
+    </div>
14
+    <div v-if="edit" class="input-group-prepend">
15
+      <input v-if="type === 'text'" class="form-control" v-model="value" />
16
+      <input v-if="type === 'number'" type="number" class="form-control" v-model="value" />
17
+      <select v-if="type === 'select'" class="form-control" @change="selectionClick">
18
+        <option>Please select</option>
19
+        <option
20
+          v-for="option in selectOptions"
21
+          :value="option[selectValue]"
22
+          :key="option[selectValue]"
23
+          >{{ option[selectText] }}</option
24
+        >
25
+      </select>
26
+      <select v-if="type === 'selectlist'" class="form-control" v-model="value">
27
+        <option v-for="item in selectOptions" :value="item" :key="item">{{ item }}</option>
28
+      </select>
29
+      <!-- <Datetime class="form-control" v-if="type === 'datetime'" v-model="value" /> -->
30
+      <span
31
+        v-if="edit"
32
+        @click="UpdateValue()"
33
+        class="input-group-text spanCursor"
34
+        style="color: #60CBEB"
35
+      >
36
+        <eva-icon name="checkmark-outline" fill="#60CBEB"></eva-icon>
37
+      </span>
38
+      <span v-if="edit" @click="Close()" class="input-group-text spanCursor" style="color: #60CBEB">
39
+        <eva-icon name="close-outline" fill="#60CBEB"></eva-icon>
40
+      </span>
41
+    </div>
42
+  </div>
43
+</template>
44
+
45
+<script>
46
+export default {
47
+  props: ['value', 'type', 'mayEdit', 'selectOptions', 'selectValue', 'selectText', 'display'],
48
+  data() {
49
+    return {
50
+      edit: false,
51
+      myDisplay: '',
52
+    };
53
+  },
54
+  methods: {
55
+    UpdateValue() {
56
+      this.edit = false;
57
+      if (this.type !== 'select' && this.value) {
58
+        this.$emit('input', this.value);
59
+        this.$emit('change', this.value);
60
+      }
61
+    },
62
+    EditClick() {
63
+      this.edit = true;
64
+    },
65
+    Close() {
66
+      this.edit = false;
67
+    },
68
+    selectionClick(item) {
69
+      if (item.target.options.selectedIndex > 0) {
70
+        if (!this.value) {
71
+          this.myDisplay = this.selectOptions[item.target.options.selectedIndex - 1][
72
+            this.selectText
73
+          ];
74
+        }
75
+        if (this.selectOptions[item.target.options.selectedIndex - 1]) {
76
+          this.$emit('input', this.selectOptions[item.target.options.selectedIndex - 1]);
77
+          this.$emit('change', this.selectOptions[item.target.options.selectedIndex - 1]);
78
+        }
79
+      }
80
+    },
81
+  },
82
+  mounted() {
83
+    if (this.value) {
84
+      // eslint-disable-next-line vue/no-side-effects-in-computed-properties
85
+      this.myDisplay = this.value;
86
+    }
87
+    if (this.value && this.selectText) {
88
+      // eslint-disable-next-line vue/no-side-effects-in-computed-properties
89
+      this.myDisplay = this.value[this.selectText];
90
+    }
91
+    if (this.display) {
92
+      this.myDisplay = this.display;
93
+    }
94
+  },
95
+  computed: {
96
+    // eslint-disable-next-line vue/return-in-computed-property
97
+    checkValue() {
98
+      if (this.value) {
99
+        // eslint-disable-next-line vue/no-side-effects-in-computed-properties
100
+        this.myDisplay = this.value;
101
+      }
102
+      if (this.value && this.selectText) {
103
+        // eslint-disable-next-line vue/no-side-effects-in-computed-properties
104
+        this.myDisplay = this.value[this.selectText];
105
+      }
106
+      if (this.display) {
107
+        // eslint-disable-next-line vue/no-side-effects-in-computed-properties
108
+        this.myDisplay = this.display;
109
+      }
110
+    },
111
+  },
112
+  watch: {
113
+    checkValue() {
114
+      return null;
115
+    },
116
+  },
117
+};
118
+</script>

+ 338
- 0
src/components/shared/gallerySlideShow.vue Parādīt failu

1
+// Source: https://github.com/KitchenStories/vue-gallery-slideshow
2
+<template>
3
+  <transition name="modal">
4
+    <div v-if="imgIndex !== null" class="vgs" @click="close">
5
+      <button type="button" class="vgs__close" @click="close">
6
+        &times;
7
+      </button>
8
+      <button v-if="isMultiple" type="button" class="vgs__prev" @click.stop="onPrev">
9
+        &lsaquo;
10
+      </button>
11
+      <div v-if="images" class="vgs__container" @click.stop="onNext">
12
+        <img class="vgs__container__img" :src="imageUrl" :alt="alt" @click.stop="onNext" />
13
+        <slot></slot>
14
+      </div>
15
+      <button v-if="isMultiple" type="button" class="vgs__next" @click.stop="onNext">
16
+        &rsaquo;
17
+      </button>
18
+      <div v-if="isMultiple" ref="gallery" class="vgs__gallery">
19
+        <div v-if="images" class="vgs__gallery__title">
20
+          {{ imgIndex + 1 }} / {{ images.length }}
21
+        </div>
22
+        <div
23
+          v-if="images"
24
+          class="vgs__gallery__container"
25
+          :style="{ transform: 'translate(' + galleryXPos + 'px, 0)' }"
26
+        >
27
+          <img
28
+            v-for="(img, i) in images"
29
+            :key="i"
30
+            class="vgs__gallery__container__img"
31
+            :src="typeof img === 'string' ? img : img.url"
32
+            :class="{ 'vgs__gallery__container__img--active': i === imgIndex }"
33
+            :alt="typeof img === 'string' ? '' : img.alt"
34
+            @click.stop="onClickThumb(img, i)"
35
+          />
36
+        </div>
37
+      </div>
38
+    </div>
39
+  </transition>
40
+</template>
41
+
42
+<script>
43
+export default {
44
+  props: {
45
+    images: {
46
+      type: Array,
47
+      required: true,
48
+    },
49
+    index: {
50
+      type: Number,
51
+      required: false,
52
+      default: null,
53
+    },
54
+  },
55
+  data() {
56
+    return {
57
+      imgIndex: this.index,
58
+      image: null,
59
+      galleryXPos: 0,
60
+      thumbnailWidth: 120,
61
+    };
62
+  },
63
+  computed: {
64
+    imageUrl() {
65
+      const img = this.images[this.imgIndex];
66
+      if (typeof img === 'string') {
67
+        return img;
68
+      }
69
+      return img.url;
70
+    },
71
+    alt() {
72
+      const img = this.images[this.imgIndex];
73
+      if (typeof img === 'object') {
74
+        return img.alt;
75
+      }
76
+
77
+      return '';
78
+    },
79
+    isMultiple() {
80
+      return this.images.length > 1;
81
+    },
82
+  },
83
+  watch: {
84
+    index(val, prev) {
85
+      this.imgIndex = val;
86
+
87
+      // updateThumbails when popup
88
+      if (prev == null && val != null) {
89
+        this.$nextTick(() => {
90
+          this.updateThumbails();
91
+        });
92
+      }
93
+    },
94
+  },
95
+  mounted() {
96
+    window.addEventListener('keydown', (e) => {
97
+      if (e.keyCode === 37) {
98
+        this.onPrev();
99
+      } else if (e.keyCode === 39) {
100
+        this.onNext();
101
+      } else if (e.keyCode === 27) {
102
+        this.close();
103
+      }
104
+    });
105
+  },
106
+  methods: {
107
+    close() {
108
+      this.imgIndex = null;
109
+      this.$emit('close');
110
+    },
111
+    onPrev() {
112
+      if (this.imgIndex === null) return;
113
+      if (this.imgIndex > 0) {
114
+        this.imgIndex--;
115
+      } else {
116
+        this.imgIndex = this.images.length - 1;
117
+      }
118
+      this.updateThumbails();
119
+    },
120
+    onNext() {
121
+      if (this.imgIndex === null) return;
122
+      if (this.imgIndex < this.images.length - 1) {
123
+        this.imgIndex++;
124
+      } else {
125
+        this.imgIndex = 0;
126
+      }
127
+      this.updateThumbails();
128
+    },
129
+    onClickThumb(image, index) {
130
+      this.imgIndex = index;
131
+      this.updateThumbails();
132
+    },
133
+    updateThumbails() {
134
+      if (!this.$refs.gallery) {
135
+        return;
136
+      }
137
+
138
+      const galleryWidth = this.$refs.gallery.clientWidth;
139
+      const currThumbsWidth = this.imgIndex * this.thumbnailWidth;
140
+      const maxThumbsWidth = this.images.length * this.thumbnailWidth;
141
+      const centerPos = Math.floor(galleryWidth / (this.thumbnailWidth * 2)) * this.thumbnailWidth;
142
+
143
+      // Prevent scrolling of images if not needed
144
+      if (maxThumbsWidth < galleryWidth) {
145
+        return;
146
+      }
147
+
148
+      if (currThumbsWidth < centerPos) {
149
+        this.galleryXPos = 0;
150
+      } else if (
151
+        currThumbsWidth
152
+        > this.images.length * this.thumbnailWidth - galleryWidth + centerPos
153
+      ) {
154
+        this.galleryXPos = -(this.images.length * this.thumbnailWidth - galleryWidth - 20);
155
+      } else {
156
+        this.galleryXPos = -(this.imgIndex * this.thumbnailWidth) + centerPos;
157
+      }
158
+    },
159
+  },
160
+};
161
+</script>
162
+
163
+<style lang="scss">
164
+$black-alpha-80: rgba(0, 0, 0, 0.8);
165
+$black: #000;
166
+$white: #fff;
167
+$radius-medium: 8px;
168
+$radius-large: 12px;
169
+// Breakpoints
170
+$screen-xs: 480px;
171
+$screen-sm: 768px;
172
+$screen-md: 992px;
173
+$screen-lg: 1200px;
174
+// So media queries don't overlap when required, provide a maximum
175
+$screen-xs-max: ($screen-sm - 1);
176
+$screen-sm-max: ($screen-md - 1);
177
+$screen-md-max: ($screen-lg - 1);
178
+@mixin respond-to($media) {
179
+  @if $media==xs {
180
+    @media (max-width: $screen-xs-max) {
181
+      @content;
182
+    }
183
+  } @else if $media==sm {
184
+    @media (min-width: $screen-sm) and (max-width: $screen-sm-max) {
185
+      @content;
186
+    }
187
+  } @else if $media==md {
188
+    @media (min-width: $screen-md) and (max-width: $screen-md-max) {
189
+      @content;
190
+    }
191
+  } @else if $media==lg {
192
+    @media (min-width: $screen-lg) {
193
+      @content;
194
+    }
195
+  }
196
+}
197
+
198
+@mixin modal-base() {
199
+  transition: opacity 0.2s ease;
200
+  position: fixed;
201
+  z-index: 9998;
202
+}
203
+
204
+@mixin modal-mask() {
205
+  @include modal-base();
206
+  top: 0;
207
+  left: 0;
208
+  width: 100%;
209
+  min-height: 100%;
210
+  height: 100vh;
211
+  background-color: $black-alpha-80;
212
+  display: table;
213
+}
214
+
215
+.vgs {
216
+  @include modal-mask();
217
+  &__close {
218
+    color: #fff;
219
+    position: absolute;
220
+    top: 0;
221
+    right: 0;
222
+    background-color: transparent;
223
+    border: none;
224
+    font-size: 25px;
225
+    width: 50px;
226
+    height: 50px;
227
+    cursor: pointer;
228
+    z-index: 999;
229
+    &:focus {
230
+      outline: 0;
231
+    }
232
+  }
233
+  &__prev,
234
+  &__next {
235
+    position: absolute;
236
+    top: 50%;
237
+    margin-top: -25px;
238
+    width: 50px;
239
+    height: 50px;
240
+    z-index: 999;
241
+    cursor: pointer;
242
+    font-size: 40px;
243
+    color: #fff;
244
+    background-color: transparent;
245
+    border: none;
246
+    &:focus {
247
+      outline: 0;
248
+    }
249
+  }
250
+  &__prev {
251
+    left: 0;
252
+  }
253
+  &__next {
254
+    right: 0;
255
+  }
256
+  &__container {
257
+    position: absolute;
258
+    overflow: hidden;
259
+    cursor: pointer;
260
+    overflow: hidden;
261
+    max-width: 100vh;
262
+    margin: 0.5rem auto 0;
263
+    left: 0.5rem;
264
+    right: 0.5rem;
265
+    height: 60vh;
266
+    border-radius: $radius-large;
267
+    background-color: $black;
268
+    @include respond-to(xs) {
269
+      width: 100%;
270
+      max-width: 100%;
271
+      top: 50%;
272
+      margin-top: -140px;
273
+      left: 0;
274
+      right: 0;
275
+      border-radius: 0;
276
+      height: 280px;
277
+    }
278
+
279
+    &__img {
280
+      width: 100%;
281
+      height: 100%;
282
+      object-fit: contain;
283
+    }
284
+  }
285
+}
286
+
287
+.vgs__gallery {
288
+  @include respond-to(xs) {
289
+    display: none;
290
+  }
291
+  overflow-x: hidden;
292
+  overflow-y: hidden;
293
+  position: absolute;
294
+  bottom: 10px;
295
+  margin: auto;
296
+  max-width: 100vh;
297
+  white-space: nowrap;
298
+  left: 0.5rem;
299
+  right: 0.5rem;
300
+  &__title {
301
+    color: $white;
302
+    margin-bottom: 0.5rem;
303
+  }
304
+  &__container {
305
+    overflow: visible;
306
+    display: block;
307
+    height: 250px;
308
+    white-space: nowrap;
309
+    transition: all 200ms ease-in-out;
310
+    width: 100%;
311
+    &__img {
312
+      width: 250px;
313
+      height: 250px;
314
+      object-fit: cover;
315
+      display: inline-block;
316
+      float: none;
317
+      margin-right: 20px;
318
+      cursor: pointer;
319
+      opacity: 0.6;
320
+      border-radius: $radius-medium;
321
+    }
322
+    &__img--active {
323
+      width: 100px;
324
+      display: inline-block;
325
+      float: none;
326
+      opacity: 1;
327
+    }
328
+  }
329
+}
330
+
331
+.modal-enter {
332
+  opacity: 0;
333
+}
334
+
335
+.modal-leave-active {
336
+  opacity: 0;
337
+}
338
+</style>

+ 51
- 11
src/components/shared/listView.vue Parādīt failu

5
     <div style="height:5px"></div>
5
     <div style="height:5px"></div>
6
     <div class="d-flex justify-content-between table-title">
6
     <div class="d-flex justify-content-between table-title">
7
       <div class="p-2" v-if="!hideSearch">
7
       <div class="p-2" v-if="!hideSearch">
8
-        <input v-model="searchItem" class="form-control" :placeholder="currentPage" />
8
+        <input v-model="searchItem" class="form-control" placeholder="Search...." />
9
       </div>
9
       </div>
10
       <div class="p-2" v-if="title">
10
       <div class="p-2" v-if="title">
11
         <h2>{{title}}</h2>
11
         <h2>{{title}}</h2>
14
         <div class="d-flex flex-row">
14
         <div class="d-flex flex-row">
15
           <div class="p2" v-if="selectedItems.length > 0">
15
           <div class="p2" v-if="selectedItems.length > 0">
16
             <div
16
             <div
17
-              class="btn btn-primary myBackground btn-width"
17
+              class="btn btn-primary myBackground btn-width cursor-pointer"
18
               @click="onClearSelected()"
18
               @click="onClearSelected()"
19
             >Clear Selected</div>
19
             >Clear Selected</div>
20
           </div>
20
           </div>
21
           <div class="p2" v-if="showNew">
21
           <div class="p2" v-if="showNew">
22
-            <div class="btn btn-primary myBackground btn-width" @click="onNew()">New</div>
22
+            <div class="btn btn-primary myBackground btn-width cursor-pointer" @click="onNew()">New</div>
23
           </div>
23
           </div>
24
         </div>
24
         </div>
25
       </div>
25
       </div>
28
     <div v-if="items && items.length > 0" class="table-responsive">
28
     <div v-if="items && items.length > 0" class="table-responsive">
29
       <table
29
       <table
30
         id="table"
30
         id="table"
31
-        :class="{'table table-bordered table-hover': (1 === 1), 'table-sm': compact}"
31
+        :class="{'table table-hover': (1 === 1), 'table-sm': compact, 'table-bordered': bordered, 'table-striped': striped}"
32
       >
32
       >
33
-        <thead class="my-table table-header">
33
+        <thead>
34
           <tr class="dnd-moved">
34
           <tr class="dnd-moved">
35
             <th v-for="(column, c) in Columns" :key="c">
35
             <th v-for="(column, c) in Columns" :key="c">
36
-              <div @click="sortBy(column)">{{ column }}</div>
36
+              <div
37
+                @click="sortBy(column)"
38
+                @mouseover="hover = c"
39
+                @mouseleave="hover = -1"
40
+                :class="{ active: hover === c }"
41
+              >
42
+                <div class="d-flex bd-highlight">
43
+                  <div class="p-2 w-100 bd-highlight">{{ column | toProper }}</div>
44
+                  <div class="p-2 flex-shrink-1 bd-highlight">
45
+                    <img
46
+                      src="../../../public/img/sort-up.png"
47
+                      height="8px;"
48
+                      v-if="sortKey === column && reverse"
49
+                    />
50
+                    <img
51
+                      src="../../../public/img/sort-down.png"
52
+                      height="8px;"
53
+                      v-if="sortKey === column && !reverse"
54
+                    />
55
+                  </div>
56
+                </div>
57
+              </div>
37
             </th>
58
             </th>
38
             <th v-if="editable"></th>
59
             <th v-if="editable"></th>
39
             <th v-if="deleteable"></th>
60
             <th v-if="deleteable"></th>
60
         </tbody>
81
         </tbody>
61
       </table>
82
       </table>
62
       <div class="d-flex justify-content-between" v-if="showPager">
83
       <div class="d-flex justify-content-between" v-if="showPager">
63
-        <div class="p-1">{{ currentPage + ' / ' + PageCount }}</div>
84
+        <div
85
+          class="p-1"
86
+        >{{ currentPage + ' / ' + PageCount + (!hideItemCount ? ' - (' + items.length + ' items)' : '')}}</div>
64
         <div class="p-1">
87
         <div class="p-1">
65
           <BasePagination
88
           <BasePagination
66
             :currentPage="currentPage"
89
             :currentPage="currentPage"
136
     deleteable: {
159
     deleteable: {
137
       default: false,
160
       default: false,
138
     },
161
     },
139
-    columnsCount: {
162
+    columnCount: {
140
       default: 6,
163
       default: 6,
141
     },
164
     },
142
     showPager: {
165
     showPager: {
148
     sortKey: {
171
     sortKey: {
149
       default: 'id',
172
       default: 'id',
150
     },
173
     },
174
+    hideItemCount: {
175
+      default: false,
176
+    },
177
+
178
+    currentPage: {
179
+      default: 1,
180
+    },
181
+    bordered: {
182
+      default: true,
183
+    },
184
+    striped: {
185
+      default: true,
186
+    },
151
   },
187
   },
152
   data() {
188
   data() {
153
     return {
189
     return {
190
+      hover: -1,
154
       selectedItems: [],
191
       selectedItems: [],
155
       showControl: false,
192
       showControl: false,
156
       reverse: false,
193
       reverse: false,
157
       searchItem: '',
194
       searchItem: '',
195
+      visibleItemsPerPageCount: 20,
158
       itemsPerPageList: ItemsPerPageList,
196
       itemsPerPageList: ItemsPerPageList,
159
-      visibleItemsPerPageCount: 0,
160
-      currentPage: 1,
161
     };
197
     };
162
   },
198
   },
163
   methods: {
199
   methods: {
257
               !list.includes(Object.keys(item)[o])
293
               !list.includes(Object.keys(item)[o])
258
               && !Array.isArray(Object.values(item)[o])
294
               && !Array.isArray(Object.values(item)[o])
259
             ) {
295
             ) {
260
-              if (list.length < this.columnsCount) {
296
+              if (list.length < this.columnCount) {
261
                 list.push(Object.keys(item)[o]);
297
                 list.push(Object.keys(item)[o]);
262
               }
298
               }
263
             }
299
             }
297
   text-decoration: none;
333
   text-decoration: none;
298
   color: #333333;
334
   color: #333333;
299
 }
335
 }
336
+.active {
337
+  background-color: rgba(255, 255, 255, 0.5);
338
+  cursor: pointer;
339
+}
300
 .table > tbody > tr > td {
340
 .table > tbody > tr > td {
301
   vertical-align: middle;
341
   vertical-align: middle;
302
 }
342
 }

+ 8
- 6
src/components/shared/navBar.vue Parādīt failu

91
                   class="dropdown-item cursor-pointer"
91
                   class="dropdown-item cursor-pointer"
92
                   @click="routerGoTo('/property/new/Rental')"
92
                   @click="routerGoTo('/property/new/Rental')"
93
                 >To Rent</a>
93
                 >To Rent</a>
94
-                <hr />
94
+                <hr v-if="isLoggedIn" />
95
                 <a
95
                 <a
96
                   v-if="isLoggedIn"
96
                   v-if="isLoggedIn"
97
                   class="dropdown-item cursor-pointer"
97
                   class="dropdown-item cursor-pointer"
98
-                  @click="routerGoTo('/property/admin/list/my')"
98
+                  @click="routerGoTo('/properties')"
99
                 >My Properties</a>
99
                 >My Properties</a>
100
               </div>
100
               </div>
101
             </li>
101
             </li>
214
 
214
 
215
 <script>
215
 <script>
216
 import { mapState, mapActions } from 'vuex';
216
 import { mapState, mapActions } from 'vuex';
217
+import Log from '../../assets/Log';
217
 
218
 
218
 export default {
219
 export default {
219
   computed: {
220
   computed: {
224
       'person',
225
       'person',
225
       'token',
226
       'token',
226
     ]),
227
     ]),
227
-    NAME() {
228
-      return this.person.name;
229
-    },
230
     isLoggedIn() {
228
     isLoggedIn() {
231
-      return this.token !== null && this.token !== undefined;
229
+      console.log(Log.isLoggedIn());
230
+      return Log.isLoggedIn();
231
+    },
232
+    NAME() {
233
+      return Log.getPerson().name;
232
     },
234
     },
233
     // eslint-disable-next-line vue/return-in-computed-property
235
     // eslint-disable-next-line vue/return-in-computed-property
234
     Logout() {
236
     Logout() {

+ 11
- 27
src/components/shared/searchTab.vue Parādīt failu

38
                   aria-controls="pills-video"
38
                   aria-controls="pills-video"
39
                   aria-selected="true"
39
                   aria-selected="true"
40
                   v-on:click="updateType('Timeshare')"
40
                   v-on:click="updateType('Timeshare')"
41
-                >Timeshare</a>
41
+                  >Timeshare</a
42
+                >
42
               </li>
43
               </li>
43
               <li class="nav-item">
44
               <li class="nav-item">
44
                 <a
45
                 <a
50
                   aria-controls="pills-plans"
51
                   aria-controls="pills-plans"
51
                   aria-selected="false"
52
                   aria-selected="false"
52
                   v-on:click="updateType('Residential')"
53
                   v-on:click="updateType('Residential')"
53
-                >Residential</a>
54
+                  >Residential</a
55
+                >
54
               </li>
56
               </li>
55
               <li class="nav-item">
57
               <li class="nav-item">
56
                 <a
58
                 <a
62
                   aria-controls="pills-map"
64
                   aria-controls="pills-map"
63
                   aria-selected="false"
65
                   aria-selected="false"
64
                   v-on:click="updateType('Commercial')"
66
                   v-on:click="updateType('Commercial')"
65
-                >Commercial</a>
67
+                  >Commercial</a
68
+                >
66
               </li>
69
               </li>
67
             </ul>
70
             </ul>
68
             <div class="tab-content" id="pills-tabContent">
71
             <div class="tab-content" id="pills-tabContent">
80
                 role="tabpanel"
83
                 role="tabpanel"
81
                 aria-labelledby="pills-plans-tab"
84
                 aria-labelledby="pills-plans-tab"
82
               >
85
               >
83
-                <propertySearch propertyType="Residential" @updateSearch="updateSearch" />
86
+                <propertySearch propertyType="Residential" />
84
               </div>
87
               </div>
85
               <div
88
               <div
86
                 class="tab-pane fade"
89
                 class="tab-pane fade"
88
                 role="tabpanel"
91
                 role="tabpanel"
89
                 aria-labelledby="pills-map-tab"
92
                 aria-labelledby="pills-map-tab"
90
               >
93
               >
91
-                <propertySearch propertyType="Commercial" @updateSearch="updateSearch" />
94
+                <propertySearch propertyType="Commercial" />
92
               </div>
95
               </div>
93
             </div>
96
             </div>
94
           </div>
97
           </div>
115
     return {
118
     return {
116
       selectedPropertyType: 'timeshare',
119
       selectedPropertyType: 'timeshare',
117
       keyword: '',
120
       keyword: '',
118
-      propertySearch: {
119
-        keyword: '',
120
-        userName: '',
121
-        salesType: 'Sale',
122
-        propertyUsageType: 'All',
123
-        propertyType: 'All',
124
-        province: 'All',
125
-        city: 'All',
126
-        suburb: 'All',
127
-        minPrice: 0,
128
-        maxPrice: 0,
129
-      },
130
     };
121
     };
131
   },
122
   },
132
   computed: {
123
   computed: {
133
     ...mapState('weekList', ['filter']),
124
     ...mapState('weekList', ['filter']),
125
+    ...mapState('propertySearch', ['propertySearch']),
134
   },
126
   },
135
   methods: {
127
   methods: {
136
     updateType(item) {
128
     updateType(item) {
137
       this.selectedPropertyType = item;
129
       this.selectedPropertyType = item;
138
     },
130
     },
139
-    updateSearch(item) {
140
-      this.propertySearch = item;
141
-      this.propertySearch.propertyUsageType = this.selectedPropertyType;
142
-      this.propertySearch.keyword = this.filter.keyword;
143
-    },
144
     Search() {
131
     Search() {
145
       if (this.selectedPropertyType === 'timeshare') {
132
       if (this.selectedPropertyType === 'timeshare') {
146
         this.$router.push('/timesharesearch');
133
         this.$router.push('/timesharesearch');
147
       } else {
134
       } else {
148
-        // this.$router.push('/property/search');
149
-        this.$router.push({
150
-          path: '/property/search',
151
-          query: this.propertySearch,
152
-        });
135
+        this.propertySearch.propertyUsageType = this.selectedPropertyType;
136
+        this.$router.push('/property/propertySearch/results');
153
       }
137
       }
154
     },
138
     },
155
   },
139
   },

+ 13
- 3
src/components/timeshare/buy/buyPage.vue Parādīt failu

35
           </div>
35
           </div>
36
           <hr />
36
           <hr />
37
           <div class="row mb-4">
37
           <div class="row mb-4">
38
-            <div class="container col-md-4">
39
-              <div class="accordion" id="accordionExample">
40
-                <div class="card" v-for="(region, r) in availRegion" :key="r">
38
+            <div class="container col-md-4" v-if="availRegion">
39
+              <div class="accordion" id="accordionExample" v-if="availRegion.length > 0">
40
+                <div class="card darker-border" v-for="(region, r) in availRegion" :key="r">
41
                   <a
41
                   <a
42
                     class="mb-0 color-text-a"
42
                     class="mb-0 color-text-a"
43
                     data-toggle="collapse"
43
                     data-toggle="collapse"
77
                     </div>
77
                     </div>
78
                   </div>
78
                   </div>
79
                 </div>
79
                 </div>
80
+                  </div>
81
+                  <div v-else>
82
+                    <Alert :text="'No items found...'" :type="'INFO'" />
80
               </div>
83
               </div>
81
             </div>
84
             </div>
82
             <div class="col-md-8">
85
             <div class="col-md-8">
110
 </template>
113
 </template>
111
 <script>
114
 <script>
112
 import { mapState, mapActions } from 'vuex';
115
 import { mapState, mapActions } from 'vuex';
116
+import Alert from '../../shared/alert.vue';
113
 
117
 
114
 export default {
118
 export default {
115
   name: 'TimeshareToBuy',
119
   name: 'TimeshareToBuy',
120
+  components:{
121
+    Alert
122
+  },
116
   data() {
123
   data() {
117
     return {
124
     return {
118
       myMap: 'SouthAfrica',
125
       myMap: 'SouthAfrica',
166
   border-radius: 5px;
173
   border-radius: 5px;
167
   cursor: pointer;
174
   cursor: pointer;
168
 }
175
 }
176
+.darker-border {
177
+  border-color: #A5A5A5;
178
+}
169
 </style>
179
 </style>

+ 38
- 10
src/components/timeshare/myWeeksPage.vue Parādīt failu

1
 <template>
1
 <template>
2
-  <div>
3
-    <br />
4
-    <br />
5
-    <br />
6
-    <br />
7
-    <br />
8
-    <div class="row">
9
-      <div class="offset-1 col-md-10">
10
-        <ListView :items="items" :editable="true" @onEdit="onEdit" />
11
-      </div>
2
+<div>
3
+  <br />
4
+  <div class="row">
5
+    <div class="offset-1 col-md-3">
6
+      {{items[selectedItems]}}
7
+      <button 
8
+        type="button"
9
+        :class="{'form-control btn btn-primary ': (1===1), 'my-disable': (!(selectedItems.length == 1) || !items[selectedItems] || items[selectedItems].status.code !== 'A2')}"
10
+        value="Publish Week"
11
+        :disabled="!(items.length == 1)"
12
+      >
13
+        Publish selected week
14
+      </button>
15
+    </div>
16
+    <div class="offset-1 col-md-10">
17
+      <ListView :items="items" :editable="true" @onNew="onNew" @onRowClick="onRowClick"
18
+          @onClearSelected="onClearSelected"  />
12
     </div>
19
     </div>
13
   </div>
20
   </div>
21
+</div>
14
 </template>
22
 </template>
15
 <script>
23
 <script>
16
 import { mapState, mapActions } from 'vuex';
24
 import { mapState, mapActions } from 'vuex';
24
   data() {
32
   data() {
25
     return {
33
     return {
26
       user: Log.getUser(),
34
       user: Log.getUser(),
35
+      selectedItems: [],
27
     };
36
     };
28
   },
37
   },
29
   mounted() {
38
   mounted() {
38
     onEdit(item) {
47
     onEdit(item) {
39
       this.$router.push(`/timeshare/${item.id}`);
48
       this.$router.push(`/timeshare/${item.id}`);
40
     },
49
     },
50
+    onNew() {
51
+      this.$router.push('/timeshare/sell');
52
+    },
53
+    onRowClick(items) {
54
+      this.selectedItems = items;
55
+    },
56
+    onClearSelected() {
57
+      this.selectedItems = [];
58
+    },
41
   },
59
   },
42
 };
60
 };
43
 </script>
61
 </script>
62
+<style scoped>
63
+.my-disable {
64
+  background-color: silver;
65
+  border-color: silver;
66
+}
67
+.my-disable:hover {
68
+  background-color: lightgray;
69
+  border-color: lightgray;
70
+}
71
+</style>

+ 9
- 0
src/main.js Parādīt failu

38
   return `R ${formatter.format(value)}`;
38
   return `R ${formatter.format(value)}`;
39
 });
39
 });
40
 
40
 
41
+Vue.filter('toProper', (value) => {
42
+  if (typeof value !== 'string') {
43
+    console.log(typeof value);
44
+    return value;
45
+  }
46
+  value = value.replace(/([a-z])([A-Z])/g, '$1 $2');
47
+  return value.charAt(0).toUpperCase() + value.slice(1);
48
+});
49
+
41
 
50
 
42
 Vue.filter('toDate', value => value.substring(0, value.length > 9 ? 10 : value.length));
51
 Vue.filter('toDate', value => value.substring(0, value.length > 9 ? 10 : value.length));
43
 
52
 

+ 231
- 218
src/router/index.js Parādīt failu

50
 import TemplatePage from '../components/communication/templatePage.vue';
50
 import TemplatePage from '../components/communication/templatePage.vue';
51
 import CarouselList from '../components/admin/misc/carouselList.vue';
51
 import CarouselList from '../components/admin/misc/carouselList.vue';
52
 import CarouselDetail from '../components/admin/misc/carousel.vue';
52
 import CarouselDetail from '../components/admin/misc/carousel.vue';
53
+import AlertPage from '../components/shared/alertPage.vue';
54
+import PropertySearchResults from '../components/property/propertySearchResults.vue';
53
 
55
 
54
 Vue.use(Router);
56
 Vue.use(Router);
55
 
57
 
61
       y: 0,
63
       y: 0,
62
     };
64
     };
63
   },
65
   },
64
-  routes: [{
65
-    path: '/',
66
-    name: 'Home',
67
-    component: HomePage,
68
-  },
69
-  {
70
-    path: '/about/us',
71
-    name: 'aboutus',
72
-    component: AboutUs,
73
-  },
74
-  {
75
-    path: '/about/timeshare',
76
-    name: 'abouttimeshare',
77
-    component: AboutTimeshare,
78
-  },
79
-  {
80
-    path: '/communication/template',
81
-    name: 'template',
82
-    component: TemplatePage,
83
-  },
84
-  {
85
-    path: '/timeshare/sell',
86
-    name: 'TimeshareSell',
87
-    component: TimeshareSell,
88
-  },
89
-  {
90
-    path: '/timeshare/buy',
91
-    name: 'TimeshareBuy',
92
-    component: TimeshareBuy,
93
-  },
94
-  {
95
-    path: '/timeshare/faq',
96
-    name: 'TimeshareFAQ',
97
-    component: TimeshareFAQ,
98
-  },
99
-  {
100
-    path: '/timeshare/myWeeks',
101
-    name: 'MyWeeks',
102
-    component: MyWeeksPage,
103
-  },
104
-  {
105
-    path: '/user/login',
106
-    name: 'Login',
107
-    component: Login,
108
-  },
109
-  {
110
-    path: '/user/updateProfileInfo',
111
-    name: 'UpdateInfo',
112
-    component: UpdateInfo,
113
-  },
114
-  {
115
-    path: '/user/register',
116
-    name: 'PrivateIndividual',
117
-    component: PrivateIndividual,
118
-  },
119
-  {
120
-    path: '/user/registeragency',
121
-    name: 'Agency',
122
-    component: Agency,
123
-  },
124
-  {
125
-    path: '/property/property/:id',
126
-    name: 'PropertyPage',
127
-    component: PropertyPage,
128
-  },
129
-  {
130
-    path: '/property/:propertyUsageType/search',
131
-    name: 'PropertySearch',
132
-    component: PropertySearch,
133
-  },
134
-  {
135
-    path: '/property/search',
136
-    name: 'PropertySearchTab',
137
-    component: PropertySearch,
138
-  },
139
-  {
140
-    path: '/property/new/:saleType',
141
-    name: 'PropertyNew',
142
-    component: PropertyCreate,
143
-  },
144
-  {
145
-    path: '/property/new/:propertyUsageType/:saleType',
146
-    name: 'PropertyNewFromSearch',
147
-    component: PropertyCreate,
148
-  },
149
-  {
150
-    path: '/property/edit',
151
-    name: 'PropertyEdit',
152
-    component: PropertyEdit,
153
-  },
154
-  {
155
-    path: '/property/admin/list/:by',
156
-    name: 'PropertyListAdmin',
157
-    component: PropertyList,
158
-  },
159
-  {
160
-    path: '/propertyTypes/list',
161
-    name: 'PropertyTypeList',
162
-    component: PropertyTypeList,
163
-  },
164
-  {
165
-    path: '/propertyType/new',
166
-    name: 'PropertyTypeNew',
167
-    component: PropertyType,
168
-  },
169
-  {
170
-    path: '/propertyType/:id',
171
-    name: 'PropertyTypeEdit',
172
-    component: PropertyType,
173
-  },
174
-  {
175
-    path: '/userDefinedGroups/list',
176
-    name: 'UserDefinedGroupsList',
177
-    component: UserDefinedGroups,
178
-  },
179
-  {
180
-    path: '/userDefinedGroups/userDefinedGroup/:id',
181
-    name: 'UserDefinedGroupEdit',
182
-    component: UserDefinedGroup,
183
-  },
184
-  {
185
-    path: '/userDefinedGroups/userDefinedGroup',
186
-    name: 'UserDefinedGroupNew',
187
-    component: UserDefinedGroup,
188
-  },
189
-  {
190
-    path: '/status/list',
191
-    name: 'StatusList',
192
-    component: Status,
193
-  },
194
-  {
195
-    path: '/status/timeshareAdmin',
196
-    name: 'TimeshareAdmin',
197
-    component: timeshareAdminPage,
198
-  },
199
-  {
200
-    path: '/status/tenderWeekAdmin',
201
-    name: 'TenderWeekAdmin',
202
-    component: tenderWeekAdminPage,
203
-  },
204
-  {
205
-    path: '/status/userManagementPage',
206
-    name: 'userManagementPage',
207
-    component: userManagementPage,
208
-  },
209
-  {
210
-    path: '/status/agentUserManagementPage',
211
-    name: 'agentManagementPage',
212
-    component: agentManagementPage,
213
-  },
214
-  {
215
-    path: '/status/changeLogPage',
216
-    name: 'changeLogPage',
217
-    component: changeLogPage,
218
-  },
219
-  {
220
-    path: '/unitConfiguration/list',
221
-    name: 'UnitConfiguration',
222
-    component: UnitConfiguration,
223
-  },
224
-  {
225
-    path: '/contactus',
226
-    name: 'ContactUs',
227
-    component: ContactUs,
228
-  },
229
-  {
230
-    path: '/privacypolicy',
231
-    name: 'PrivacyPolicy',
232
-    component: PrivacyPolicy,
233
-  },
234
-  {
235
-    path: '/resort/:resortCode',
236
-    name: 'ResortPage',
237
-    component: ResortPage,
238
-    props: true,
239
-  },
240
-  {
241
-    path: '/resort/:resortCode/:unitNumber',
242
-    name: 'UnitPage',
243
-    component: UnitPage,
244
-    props: true,
245
-  },
246
-  {
247
-    path: '/timeshare/:weekId',
248
-    name: 'TimeshareSell',
249
-    component: TimeshareSell,
250
-    props: true,
251
-  },
252
-  {
253
-    path: '/MakeOffer',
254
-    name: 'MakeOffer',
255
-    component: MakeOffer,
256
-  },
257
-  {
258
-    path: '/Offers',
259
-    name: 'Offers',
260
-    component: Offer,
261
-  },
262
-  {
263
-    path: '/timesharesearch',
264
-    name: 'TimeshareSearch',
265
-    component: TimeshareSearch,
266
-  },
267
-  {
268
-    path: '/searchLog',
269
-    name: 'SearchLog',
270
-    component: searchLog,
271
-  },
272
-  {
273
-    path: '/carousel',
274
-    name: 'carousel',
275
-    component: CarouselList,
276
-  },
277
-  {
278
-    path: '/carousel/details/:id',
279
-    name: 'CarouselDetails',
280
-    component: CarouselDetail,
281
-  },
66
+  routes: [
67
+    {
68
+      path: '/',
69
+      name: 'Home',
70
+      component: HomePage,
71
+    },
72
+    {
73
+      path: '/shared/alert',
74
+      name: 'AlertPage',
75
+      component: AlertPage,
76
+    },
77
+    {
78
+      path: '/about/us',
79
+      name: 'aboutus',
80
+      component: AboutUs,
81
+    },
82
+    {
83
+      path: '/about/timeshare',
84
+      name: 'abouttimeshare',
85
+      component: AboutTimeshare,
86
+    },
87
+    {
88
+      path: '/communication/template',
89
+      name: 'template',
90
+      component: TemplatePage,
91
+    },
92
+    {
93
+      path: '/timeshare/sell',
94
+      name: 'TimeshareSell',
95
+      component: TimeshareSell,
96
+    },
97
+    {
98
+      path: '/timeshare/buy',
99
+      name: 'TimeshareBuy',
100
+      component: TimeshareBuy,
101
+    },
102
+    {
103
+      path: '/timeshare/faq',
104
+      name: 'TimeshareFAQ',
105
+      component: TimeshareFAQ,
106
+    },
107
+    {
108
+      path: '/timeshare/myWeeks',
109
+      name: 'MyWeeks',
110
+      component: MyWeeksPage,
111
+    },
112
+    {
113
+      path: '/user/login',
114
+      name: 'Login',
115
+      component: Login,
116
+    },
117
+    {
118
+      path: '/user/updateProfileInfo',
119
+      name: 'UpdateInfo',
120
+      component: UpdateInfo,
121
+    },
122
+    {
123
+      path: '/user/register',
124
+      name: 'PrivateIndividual',
125
+      component: PrivateIndividual,
126
+    },
127
+    {
128
+      path: '/user/registeragency',
129
+      name: 'Agency',
130
+      component: Agency,
131
+    },
132
+    {
133
+      path: '/property/property/:id',
134
+      name: 'PropertyPage',
135
+      component: PropertyPage,
136
+    },
137
+    {
138
+      path: '/property/:propertyUsageType/search',
139
+      name: 'PropertySearch',
140
+      component: PropertySearch,
141
+    },
142
+    {
143
+      path: '/property/search',
144
+      name: 'PropertySearchTab',
145
+      component: PropertySearch,
146
+    },
147
+    {
148
+      path: '/property/new/:saleType',
149
+      name: 'PropertyNew',
150
+      component: PropertyCreate,
151
+    },
152
+    {
153
+      path: '/property/new/:propertyUsageType/:saleType',
154
+      name: 'PropertyNewFromSearch',
155
+      component: PropertyCreate,
156
+    },
157
+    {
158
+      path: '/property/edit',
159
+      name: 'PropertyEdit',
160
+      component: PropertyEdit,
161
+    },
162
+    {
163
+      path: '/properties',
164
+      name: 'PropertyListAdmin',
165
+      component: PropertyList,
166
+    },
167
+    {
168
+      path: '/propertyTypes/list',
169
+      name: 'PropertyTypeList',
170
+      component: PropertyTypeList,
171
+    },
172
+    {
173
+      path: '/propertyType/new',
174
+      name: 'PropertyTypeNew',
175
+      component: PropertyType,
176
+    },
177
+    {
178
+      path: '/propertyType/:id',
179
+      name: 'PropertyTypeEdit',
180
+      component: PropertyType,
181
+    },
182
+    {
183
+      path: '/userDefinedGroups/list',
184
+      name: 'UserDefinedGroupsList',
185
+      component: UserDefinedGroups,
186
+    },
187
+    {
188
+      path: '/userDefinedGroups/userDefinedGroup/:id',
189
+      name: 'UserDefinedGroupEdit',
190
+      component: UserDefinedGroup,
191
+    },
192
+    {
193
+      path: '/userDefinedGroups/userDefinedGroup',
194
+      name: 'UserDefinedGroupNew',
195
+      component: UserDefinedGroup,
196
+    },
197
+    {
198
+      path: '/status/list',
199
+      name: 'StatusList',
200
+      component: Status,
201
+    },
202
+    {
203
+      path: '/status/timeshareAdmin',
204
+      name: 'TimeshareAdmin',
205
+      component: timeshareAdminPage,
206
+    },
207
+    {
208
+      path: '/status/tenderWeekAdmin',
209
+      name: 'TenderWeekAdmin',
210
+      component: tenderWeekAdminPage,
211
+    },
212
+    {
213
+      path: '/status/userManagementPage',
214
+      name: 'userManagementPage',
215
+      component: userManagementPage,
216
+    },
217
+    {
218
+      path: '/status/agentUserManagementPage',
219
+      name: 'agentManagementPage',
220
+      component: agentManagementPage,
221
+    },
222
+    {
223
+      path: '/status/changeLogPage',
224
+      name: 'changeLogPage',
225
+      component: changeLogPage,
226
+    },
227
+    {
228
+      path: '/unitConfiguration/list',
229
+      name: 'UnitConfiguration',
230
+      component: UnitConfiguration,
231
+    },
232
+    {
233
+      path: '/contactus',
234
+      name: 'ContactUs',
235
+      component: ContactUs,
236
+    },
237
+    {
238
+      path: '/privacypolicy',
239
+      name: 'PrivacyPolicy',
240
+      component: PrivacyPolicy,
241
+    },
242
+    {
243
+      path: '/resort/:resortCode',
244
+      name: 'ResortPage',
245
+      component: ResortPage,
246
+      props: true,
247
+    },
248
+    {
249
+      path: '/resort/:resortCode/:unitNumber',
250
+      name: 'UnitPage',
251
+      component: UnitPage,
252
+      props: true,
253
+    },
254
+    {
255
+      path: '/timeshare/:weekId',
256
+      name: 'TimeshareSellEdit',
257
+      component: TimeshareSell,
258
+      props: true,
259
+    },
260
+    {
261
+      path: '/MakeOffer',
262
+      name: 'MakeOffer',
263
+      component: MakeOffer,
264
+    },
265
+    {
266
+      path: '/Offers',
267
+      name: 'Offers',
268
+      component: Offer,
269
+    },
270
+    {
271
+      path: '/timesharesearch',
272
+      name: 'TimeshareSearch',
273
+      component: TimeshareSearch,
274
+    },
275
+    {
276
+      path: '/searchLog',
277
+      name: 'SearchLog',
278
+      component: searchLog,
279
+    },
280
+    {
281
+      path: '/carousel',
282
+      name: 'carousel',
283
+      component: CarouselList,
284
+    },
285
+    {
286
+      path: '/carousel/details/:id',
287
+      name: 'CarouselDetails',
288
+      component: CarouselDetail,
289
+    },
290
+    {
291
+      path: '/property/propertySearch/results',
292
+      name: 'PropertySearchResults',
293
+      component: PropertySearchResults,
294
+    },
282
   ],
295
   ],
283
 });
296
 });

+ 4
- 0
src/store/index.js Parādīt failu

25
 import PropertyEdit from './modules/property/propertyEdit';
25
 import PropertyEdit from './modules/property/propertyEdit';
26
 import Carousel from './modules/misc/carousel';
26
 import Carousel from './modules/misc/carousel';
27
 import Individual from './modules/user/individual';
27
 import Individual from './modules/user/individual';
28
+import Alert from './modules/misc/alert';
29
+import TenderWeek from './modules/timeshare/tenderWeeks';
28
 
30
 
29
 Vue.use(Vuex);
31
 Vue.use(Vuex);
30
 /* eslint no-param-reassign: ["error", { "props": false }] */
32
 /* eslint no-param-reassign: ["error", { "props": false }] */
54
     propertyEdit: PropertyEdit,
56
     propertyEdit: PropertyEdit,
55
     carousel: Carousel,
57
     carousel: Carousel,
56
     individual: Individual,
58
     individual: Individual,
59
+    alert: Alert,
60
+    tenderWeek: TenderWeek
57
   },
61
   },
58
 });
62
 });

+ 21
- 0
src/store/modules/misc/alert.js Parādīt failu

1
+export default {
2
+  namespaced: true,
3
+  state: {
4
+    type: undefined,
5
+    message: undefined,
6
+  },
7
+  mutations: {
8
+    setSuccess(state, msg) {
9
+      state.type = 'SUCCESS';
10
+      state.message = msg;
11
+    },
12
+  },
13
+  getters: {},
14
+  actions: {
15
+    success({
16
+      commit,
17
+    }, msg) {
18
+      commit('setSuccess', msg);
19
+    },
20
+  },
21
+};

+ 25
- 0
src/store/modules/misc/carousel.js Parādīt failu

4
   namespaced: true,
4
   namespaced: true,
5
   state: {
5
   state: {
6
     carouselList: [],
6
     carouselList: [],
7
+    carousel: {
8
+      id: 0,
9
+      propertyID: 0,
10
+      timeshareID: 0,
11
+      header: '',
12
+      image: '',
13
+    },
7
   },
14
   },
8
   mutations: {
15
   mutations: {
16
+    setCarouselItem(state, item) {
17
+      state.carousel = item;
18
+    },
9
     setCarouselList(state, items) {
19
     setCarouselList(state, items) {
10
       state.carouselList = items;
20
       state.carouselList = items;
11
     },
21
     },
22
+    addToCarouselList(state, item) {
23
+      state.carouselList.push(item);
24
+    },
12
     removeCarousel(state, id) {
25
     removeCarousel(state, id) {
13
       state.carouselList.pop(state.carouselList.find(p => p.id === id));
26
       state.carouselList.pop(state.carouselList.find(p => p.id === id));
14
     },
27
     },
15
   },
28
   },
16
   getters: {},
29
   getters: {},
17
   actions: {
30
   actions: {
31
+    getCarouselItem({ commit }, id) {
32
+      axios
33
+        .get(`/api/Carousel/${id}`)
34
+        .then(result => commit('setCarouselItem', result.data))
35
+        .catch(console.error);
36
+    },
18
     getCarouselList({ commit }) {
37
     getCarouselList({ commit }) {
19
       axios
38
       axios
20
         .get('/api/Carousel')
39
         .get('/api/Carousel')
21
         .then(result => commit('setCarouselList', result.data))
40
         .then(result => commit('setCarouselList', result.data))
22
         .catch(console.error);
41
         .catch(console.error);
23
     },
42
     },
43
+    saveCarouselItem({ commit }, item) {
44
+      axios
45
+        .post('/api/Carousel', item)
46
+        .then(result => commit('addToCarouselList', result.data))
47
+        .catch(console.error);
48
+    },
24
     deleteCarousel({ commit }, id) {
49
     deleteCarousel({ commit }, id) {
25
       axios
50
       axios
26
         .delete(`/api/Carousel/${id}`)
51
         .delete(`/api/Carousel/${id}`)

+ 17
- 4
src/store/modules/property/property.js Parādīt failu

148
         .catch(console.error);
148
         .catch(console.error);
149
     },
149
     },
150
     saveProperty({ commit }, item) {
150
     saveProperty({ commit }, item) {
151
-      axios
152
-        .post('/api/Property', item)
153
-        .then(result => commit('updateCurrentProperty', result.data))
154
-        .catch(console.error);
151
+      return new Promise((resolve, reject) => {
152
+        axios
153
+          .post('/api/Property', item)
154
+          .then((resp) => {
155
+            commit('updateCurrentProperty', resp);
156
+            resolve(resp);
157
+          })
158
+          .catch(() => {
159
+            reject(console.error);
160
+          });
161
+      });
155
     },
162
     },
156
     clearProperty({ commit }) {
163
     clearProperty({ commit }) {
157
       commit('clearProperty');
164
       commit('clearProperty');
183
         .then(response => commit('setPropertyFields', response.data))
190
         .then(response => commit('setPropertyFields', response.data))
184
         .catch(console.error);
191
         .catch(console.error);
185
     },
192
     },
193
+    getSavedPropertyImages({ commit }, id) {
194
+      axios
195
+        .get(`/api/PropertyImage/GetProperySavedImages/${id}`)
196
+        .then(result => commit('setPropertyImages', result.data))
197
+        .catch(console.error);
198
+    },
186
   },
199
   },
187
 };
200
 };

+ 11
- 8
src/store/modules/property/propertyEdit.js Parādīt failu

123
         .catch(console.error);
123
         .catch(console.error);
124
     },
124
     },
125
     updateProperty({ commit }, item) {
125
     updateProperty({ commit }, item) {
126
-      axios
127
-        .post('/api/PropertyImage', item.images)
128
-        .then(commit('ClearNewImages'))
129
-        .catch(console.error);
126
+      return new Promise((resolve) => {
127
+        axios
128
+          .post('/api/PropertyImage', item.images)
129
+          .then(commit('ClearNewImages'))
130
+          .catch(console.error);
130
 
131
 
131
-      axios
132
-        .put('/api/Property', item.property)
133
-        .then(commit('setProperty', item.property))
134
-        .catch(console.error);
132
+        axios
133
+          .put('/api/Property', item.property)
134
+          .then(commit('setProperty', item.property))
135
+          .catch(console.error);
136
+        resolve(true);
137
+      });
135
     },
138
     },
136
     mayEditProperty({ commit }, id) {
139
     mayEditProperty({ commit }, id) {
137
       axios
140
       axios

+ 41
- 0
src/store/modules/property/propertySearch.js Parādīt failu

19
     },
19
     },
20
     properties: [],
20
     properties: [],
21
     latestProperties: [],
21
     latestProperties: [],
22
+    suburbs: [],
23
+    searchText: '',
24
+    suburbList: [],
22
   },
25
   },
23
   mutations: {
26
   mutations: {
24
     updateSearch(state, propertySearch) {
27
     updateSearch(state, propertySearch) {
34
     onClearFilter(state, filter) {
37
     onClearFilter(state, filter) {
35
       state.propertySearch[filter] = 'All';
38
       state.propertySearch[filter] = 'All';
36
     },
39
     },
40
+    setSuburbs(state, items) {
41
+      state.suburbList = [];
42
+      state.suburbs = items;
43
+      // eslint-disable-next-line no-plusplus
44
+      for (let i = 0; i < state.suburbs.length; i++) {
45
+        state.suburbList.push(state.suburbs[i].display);
46
+      }
47
+    },
48
+    setFilter(state, value) {
49
+      state.searchText = value;
50
+    },
51
+  },
52
+  getters: {
53
+    filterSuburbs: (state) => {
54
+      let subs = state.suburbs;
55
+      if (state.searchText) {
56
+        subs = _.filter(subs, s => s.display.contains(state.searchText));
57
+      }
58
+      return subs;
59
+    },
37
   },
60
   },
38
   actions: {
61
   actions: {
39
     clearFilter({ commit }, filter) {
62
     clearFilter({ commit }, filter) {
60
       if (item.userName === '') {
83
       if (item.userName === '') {
61
         item.userName = 'Unknown';
84
         item.userName = 'Unknown';
62
       }
85
       }
86
+      if (item.suburb === '') {
87
+        item.suburb = 'All';
88
+      }
89
+      if (item.city === '') {
90
+        item.city = 'All';
91
+      }
92
+      if (item.province === '') {
93
+        item.province = 'All';
94
+      }
63
       axios
95
       axios
64
         .get(
96
         .get(
65
           `/api/Property/Search/${item.userName}/${item.keyword}/${item.salesType}/${item.propertyUsageType}/${item.propertyType}/${item.province}/${item.city}/${item.suburb}/${item.minPrice}/${item.maxPrice}`,
97
           `/api/Property/Search/${item.userName}/${item.keyword}/${item.salesType}/${item.propertyUsageType}/${item.propertyType}/${item.province}/${item.city}/${item.suburb}/${item.minPrice}/${item.maxPrice}`,
73
         .then(response => commit('setLatestProperties', response.data))
105
         .then(response => commit('setLatestProperties', response.data))
74
         .catch(console.error);
106
         .catch(console.error);
75
     },
107
     },
108
+    getSuburbs({ commit }) {
109
+      axios
110
+        .get('/api/suburb/GetSearchList')
111
+        .then(response => commit('setSuburbs', response.data))
112
+        .catch(console.error);
113
+    },
114
+    applyFilter({ commit }, value) {
115
+      commit('setFilter', { value });
116
+    },
76
   },
117
   },
77
 };
118
 };

+ 20
- 2
src/store/modules/timeshare/myWeeks.js Parādīt failu

1
+/* eslint-disable no-restricted-syntax */
2
+/* eslint-disable guard-for-in */
1
 import axios from 'axios';
3
 import axios from 'axios';
2
 
4
 
3
 export default {
5
 export default {
4
   namespaced: true,
6
   namespaced: true,
5
   state: {
7
   state: {
6
-    test: 'kobus',
7
     items: [],
8
     items: [],
8
   },
9
   },
9
   mutations: {
10
   mutations: {
11
       state.items = list;
12
       state.items = list;
12
     },
13
     },
13
   },
14
   },
14
-  getters: {},
15
+  getters: {
16
+    getNeedsVerify(state) {
17
+      return _.filter(state.items, x => x.status && x.status.code === 'A1');
18
+    },
19
+  },
15
   actions: {
20
   actions: {
16
     getItems({
21
     getItems({
17
       commit,
22
       commit,
21
           .error,
26
           .error,
22
       );
27
       );
23
     },
28
     },
29
+    verifyWeek({
30
+      commit,
31
+    }, id) {
32
+      try {
33
+        axios.post(`/api/timeshareweek/verifyweek/${id}`).catch(console.error);
34
+      } catch (err) {
35
+        console.log(err);
36
+      }
37
+    },
38
+    publishWeek(id) {
39
+      axios.post(`/api/timeshareweek/publishweek/${id}`).catch(console.error);
40
+    }
41
+
24
   },
42
   },
25
 };
43
 };

+ 22
- 0
src/store/modules/timeshare/tenderWeeks.js Parādīt failu

1
+import axios from 'axios';
2
+
3
+export default {
4
+  namespaced: true,
5
+  state: {
6
+    items: [],
7
+  },
8
+  mutations: {
9
+    setItems(state, list) {
10
+      state.items = list;
11
+    },
12
+  },
13
+  getters: {},
14
+  actions: {
15
+    getItems({ commit }) {
16
+      axios
17
+        .get('/api/timeshareweek/getTenderWeeks')
18
+        .then(result => commit('setItems', result.data))
19
+        .catch(console.error);
20
+    },
21
+  },
22
+};

Notiek ielāde…
Atcelt
Saglabāt