Subversion Repositories SmartDukaan

Rev

Rev 35957 | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
24417 govind 1
package com.spice.profitmandi.web.controller;
2
 
3
import com.spice.profitmandi.common.exception.ProfitMandiBusinessException;
4
import com.spice.profitmandi.common.model.CustomRetailer;
32812 shampa 5
import com.spice.profitmandi.common.model.ProfitMandiConstants;
24417 govind 6
import com.spice.profitmandi.dao.entity.auth.AuthUser;
27690 amit.gupta 7
import com.spice.profitmandi.dao.entity.cs.*;
36040 ranu 8
import com.spice.profitmandi.dao.entity.cs.TicketReadStatus.UserType;
27270 tejbeer 9
import com.spice.profitmandi.dao.entity.dtr.Document;
24417 govind 10
import com.spice.profitmandi.dao.entity.fofo.ActivityType;
11
import com.spice.profitmandi.dao.enumuration.cs.EscalationType;
24699 govind 12
import com.spice.profitmandi.dao.enumuration.cs.TicketStatus;
25570 tejbeer 13
import com.spice.profitmandi.dao.model.CreatePositionModel;
24417 govind 14
import com.spice.profitmandi.dao.repository.auth.AuthRepository;
27690 amit.gupta 15
import com.spice.profitmandi.dao.repository.cs.*;
27270 tejbeer 16
import com.spice.profitmandi.dao.repository.dtr.DocumentRepository;
25570 tejbeer 17
import com.spice.profitmandi.dao.repository.dtr.FofoStoreRepository;
24417 govind 18
import com.spice.profitmandi.service.authentication.RoleManager;
36040 ranu 19
import com.spice.profitmandi.service.mail.MailOutboxService;
24417 govind 20
import com.spice.profitmandi.service.user.RetailerService;
21
import com.spice.profitmandi.web.model.LoginDetails;
22
import com.spice.profitmandi.web.util.CookiesProcessor;
23
import com.spice.profitmandi.web.util.MVCResponseSender;
27690 amit.gupta 24
import org.apache.logging.log4j.LogManager;
25
import org.apache.logging.log4j.Logger;
26
import org.springframework.beans.factory.annotation.Autowired;
27
import org.springframework.mail.javamail.JavaMailSender;
28
import org.springframework.stereotype.Controller;
29
import org.springframework.transaction.annotation.Transactional;
30
import org.springframework.ui.Model;
31
import org.springframework.web.bind.annotation.*;
24417 govind 32
 
27690 amit.gupta 33
import javax.servlet.http.HttpServletRequest;
36040 ranu 34
import javax.swing.*;
27690 amit.gupta 35
import java.time.LocalDateTime;
36
import java.util.*;
37
import java.util.stream.Collectors;
38
 
24417 govind 39
@Controller
40
@Transactional(rollbackFor = Throwable.class)
41
public class CsController {
42
 
31762 tejbeer 43
    private static final Logger LOGGER = LogManager.getLogger(CsController.class);
44
    private static final String ACTIVITY_SUBJECT = "Message related ticketId#%s";
45
    private static final String PARTNER_RESOLVED_TICKET_MAIL = "Dear Partner , we have resolved your ticket # %s , request to kindly accept the same. In case you still have any concerns regarding the same pls click on %s so that we can help you.Regards\nSmartdukaan";
46
    private static final String PARTNER_REOPEN = "Dear Partner , Your ticket # %s has been re-opened as per your confirmation & we are committed to resolve it on priority.Regards\nSmartdukaan";
47
    private static final String INTERNAL_REOPEN_MAIL = "Team, Pls note that the Ticket Id %s has been re-opened by %s , pls respond on priority";
48
    private static final String INTERNAL_REOPEN_ACTIVITY_MESSAGE = "Hi,My ticket is not resolved yet,so I have reopened it";
24699 govind 49
 
31762 tejbeer 50
    @Autowired
51
    JavaMailSender mailSender;
24620 govind 52
 
31762 tejbeer 53
    @Autowired
35957 amit 54
    MailOutboxService mailOutboxService;
55
 
56
    @Autowired
31762 tejbeer 57
    private CsService csService;
24417 govind 58
 
31762 tejbeer 59
    @Autowired
60
    private CookiesProcessor cookiesProcessor;
24417 govind 61
 
31762 tejbeer 62
    @Autowired
63
    private TicketCategoryRepository ticketCategoryRepository;
24417 govind 64
 
31762 tejbeer 65
    @Autowired
66
    private TicketSubCategoryRepository ticketSubCategoryRepository;
24417 govind 67
 
31762 tejbeer 68
    @Autowired
69
    private RegionRepository regionRepository;
24417 govind 70
 
31762 tejbeer 71
    @Autowired
72
    private RetailerService retailerService;
24417 govind 73
 
31762 tejbeer 74
    @Autowired
75
    private MVCResponseSender mvcResponseSender;
24417 govind 76
 
31762 tejbeer 77
    @Autowired
78
    private AuthRepository authRepository;
24417 govind 79
 
31762 tejbeer 80
    @Autowired
81
    private PositionRepository positionRepository;
24417 govind 82
 
31762 tejbeer 83
    @Autowired
84
    private TicketRepository ticketRepository;
24417 govind 85
 
31762 tejbeer 86
    @Autowired
87
    private RoleManager roleManager;
24417 govind 88
 
31762 tejbeer 89
    @Autowired
90
    private ActivityRepository activityRepository;
24417 govind 91
 
31762 tejbeer 92
    @Autowired
93
    private ActivityAttachmentRepository activityAttachmentRepository;
27270 tejbeer 94
 
31762 tejbeer 95
    @Autowired
96
    private TicketAssignedRepository ticketAssignedRepository;
24569 govind 97
 
31762 tejbeer 98
    @Autowired
99
    private PartnerRegionRepository partnerRegionRepository;
24500 govind 100
 
31762 tejbeer 101
    @Autowired
32493 amit.gupta 102
    PartnerPositionRepository partnerPositionRepository;
25570 tejbeer 103
 
31762 tejbeer 104
    @Autowired
105
    FofoStoreRepository fofoStoreRepository;
25570 tejbeer 106
 
31762 tejbeer 107
    @Autowired
108
    DocumentRepository documentRepository;
27270 tejbeer 109
 
31762 tejbeer 110
    @GetMapping(value = "/cs/createCategory")
111
    public String getCreateCategory(HttpServletRequest request, Model model) {
112
        List<TicketCategory> ticketCategories = ticketCategoryRepository.selectAll();
113
        model.addAttribute("ticketCategories", ticketCategories);
114
        return "create-ticket-category";
115
    }
24417 govind 116
 
31762 tejbeer 117
    @PostMapping(value = "/cs/createCategory")
33081 ranu 118
    public String createCategory(HttpServletRequest request,
119
                                 @RequestParam(name = "name") String name,
120
                                 @RequestParam(name = "categoryType") int categoryType,
121
                                 @RequestParam(name = "description") String description,
122
                                 Model model) throws ProfitMandiBusinessException {
31762 tejbeer 123
        TicketCategory ticketCategory = ticketCategoryRepository.selectByName(name);
124
        if (ticketCategory != null) {
125
            throw new ProfitMandiBusinessException("name", name, "already exists!");
126
        }
33081 ranu 127
 
31762 tejbeer 128
        ticketCategory = new TicketCategory();
129
        ticketCategory.setName(name);
130
        ticketCategory.setDescription(description);
33081 ranu 131
 
132
        ticketCategory.setCategoryType(categoryType == 1);
31762 tejbeer 133
        ticketCategoryRepository.persist(ticketCategory);
134
        return "create-ticket-category";
135
    }
24417 govind 136
 
33081 ranu 137
 
31762 tejbeer 138
    @GetMapping(value = "/cs/createSubCategory")
139
    public String getCreateSubCategory(HttpServletRequest request, Model model) {
140
        List<TicketCategory> ticketCategories = ticketCategoryRepository.selectAll();
141
        model.addAttribute("ticketCategories", ticketCategories);
142
        return "create-ticket-sub-category";
143
    }
24417 govind 144
 
31762 tejbeer 145
    @GetMapping(value = "/cs/getSubCategoryByCategoryId")
146
    public String getSubCategoryByCategoryId(HttpServletRequest request, @RequestParam(name = "ticketCategoryId", defaultValue = "") int ticketCategoryId, Model model) {
147
        List<TicketSubCategory> ticketSubCategories = ticketSubCategoryRepository.selectAll(ticketCategoryId);
148
        TicketCategory ticketCategory = ticketCategoryRepository.selectById(ticketCategoryId);
33081 ranu 149
        LOGGER.info("ticketSubCategories {}", ticketSubCategories);
150
        LOGGER.info("ticketCategory {}", ticketCategory);
31762 tejbeer 151
        model.addAttribute("ticketSubCategories", ticketSubCategories);
152
        model.addAttribute("ticketCategory", ticketCategory);
153
        return "ticket-sub-category";
154
    }
24417 govind 155
 
31762 tejbeer 156
    @PostMapping(value = "/cs/createSubCategory")
157
    public String createSubCategory(HttpServletRequest request, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, @RequestParam(name = "name") String name, @RequestParam(name = "description") String description, Model model) throws ProfitMandiBusinessException {
24417 govind 158
 
31762 tejbeer 159
        TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectTicketSubCategory(categoryId, name);
160
        if (ticketSubCategory != null) {
161
            throw new ProfitMandiBusinessException("name & categoryId", name + "  " + categoryId, "already exists!");
162
        }
24417 govind 163
 
31762 tejbeer 164
        ticketSubCategory = new TicketSubCategory();
165
        ticketSubCategory.setCategoryId(categoryId);
166
        ticketSubCategory.setName(name);
167
        ticketSubCategory.setDescription(description);
168
        ticketSubCategoryRepository.persist(ticketSubCategory);
169
        return "create-ticket-sub-category";
170
    }
24417 govind 171
 
31762 tejbeer 172
    @GetMapping(value = "/cs/createRegion")
173
    public String createRegion(HttpServletRequest request, Model model) {
174
        List<Region> regions = regionRepository.selectAll();
175
        model.addAttribute("regions", regions);
176
        return "create-region";
177
    }
24417 govind 178
 
31762 tejbeer 179
    @PostMapping(value = "/cs/createRegion")
180
    public String createRegion(HttpServletRequest request, @RequestParam(name = "name") String name, @RequestParam(name = "description") String description, Model model) throws Exception {
181
        Region region = regionRepository.selectByName(name);
182
        if (region != null) {
183
            throw new ProfitMandiBusinessException("name", name, "already exists!");
184
        }
185
        region = new Region();
186
        region.setName(name);
187
        region.setDescription(description);
188
        regionRepository.persist(region);
189
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
190
        return "response";
191
    }
24417 govind 192
 
31762 tejbeer 193
    @GetMapping(value = "/cs/getPartners")
33244 ranu 194
    public String getPartners(HttpServletRequest request, @RequestParam(name = "regionId", defaultValue = "0") int regionId, Model model) throws ProfitMandiBusinessException {
31762 tejbeer 195
        List<Integer> fofoIds = fofoStoreRepository.selectAll().stream().map(x -> x.getId()).collect(Collectors.toList());
196
        List<Integer> addedfofoIds = partnerRegionRepository.selectByRegionId(regionId).stream().map(x -> x.getFofoId()).collect(Collectors.toList());
30426 tejbeer 197
 
31762 tejbeer 198
        Map<Integer, CustomRetailer> customRetailerMap = retailerService.getAllFofoRetailers();
199
        Map<Integer, CustomRetailer> fofoRetailers = fofoIds.stream().map(x -> customRetailerMap.get(x)).filter(x -> x != null).collect(Collectors.toList()).stream().collect(Collectors.toMap(x -> x.getPartnerId(), x -> x));
200
        model.addAttribute("fofoRetailers", fofoRetailers);
201
        model.addAttribute("addedfofoIds", addedfofoIds);
202
        return "added-region-partners";
203
    }
24569 govind 204
 
31762 tejbeer 205
    @GetMapping(value = "/cs/getPartnersByRegion")
33244 ranu 206
    public String getPartnersByRegion(HttpServletRequest request, @RequestParam(name = "regionId", defaultValue = "0") int regionId, Model model) throws ProfitMandiBusinessException {
31762 tejbeer 207
        List<Integer> fofoIds = null;
208
        fofoIds = partnerRegionRepository.selectByRegionId(regionId).stream().map(x -> x.getFofoId()).collect(Collectors.toList());
25570 tejbeer 209
 
31762 tejbeer 210
        if (fofoIds.contains(0)) {
211
            fofoIds = fofoStoreRepository.selectAll().stream().filter(x -> x.isActive()).collect(Collectors.toList()).stream().map(x -> x.getId()).collect(Collectors.toList());
25570 tejbeer 212
 
31762 tejbeer 213
        }
214
        Map<Integer, CustomRetailer> customRetailerMap = retailerService.getAllFofoRetailers();
25570 tejbeer 215
 
31762 tejbeer 216
        Map<Integer, CustomRetailer> fofoRetailers = fofoIds.stream().map(x -> customRetailerMap.get(x)).filter(x -> x != null).collect(Collectors.toList()).stream().collect(Collectors.toMap(x -> x.getPartnerId(), x -> x));
217
        model.addAttribute("fofoRetailers", fofoRetailers);
218
        return "added-subregion-partners";
219
    }
25570 tejbeer 220
 
31762 tejbeer 221
    @GetMapping(value = "/cs/createPartnerRegion")
222
    public String createPartnerRegion(HttpServletRequest request, Model model) {
223
        List<Region> regions = regionRepository.selectAll();
224
        model.addAttribute("regions", regions);
225
        return "create-partner-region";
226
    }
24417 govind 227
 
31762 tejbeer 228
    @PostMapping(value = "/cs/createPartnerRegion")
229
    public String createPartnerRegion(HttpServletRequest request, @RequestParam(name = "regionId") int regionId, @RequestBody List<Integer> selectedFofoIds, Model model) throws Exception {
230
        partnerRegionRepository.delete(regionId);
231
        LOGGER.info("successfully removed");
232
        LOGGER.info(selectedFofoIds.size());
233
        csService.addPartnerToRegion(regionId, selectedFofoIds);
234
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
235
        return "response";
236
    }
24417 govind 237
 
31762 tejbeer 238
    @GetMapping(value = "/cs/getPosition")
33244 ranu 239
    public String getPosition(HttpServletRequest request, @RequestParam int positionId, Model model) throws ProfitMandiBusinessException {
27410 tejbeer 240
 
31762 tejbeer 241
        Position position = positionRepository.selectById(positionId);
27410 tejbeer 242
 
31762 tejbeer 243
        List<CustomRetailer> positionIdCustomRetailer = csService.getPositionCustomRetailerMap(Arrays.asList(position)).get(position.getId());
27410 tejbeer 244
 
31762 tejbeer 245
        Map<Integer, CustomRetailer> regionRetailerMap = csService.getRegionPartners(Arrays.asList(position)).get(position.getRegionId()).stream().collect(Collectors.toMap(x -> x.getPartnerId(), x -> x));
27410 tejbeer 246
 
31762 tejbeer 247
        model.addAttribute("position", position);
248
        model.addAttribute("regionRetailerMap", regionRetailerMap);
249
        model.addAttribute("positionIdCustomRetailer", positionIdCustomRetailer);
27410 tejbeer 250
 
31762 tejbeer 251
        return "position-partner";
252
    }
27410 tejbeer 253
 
31762 tejbeer 254
    @GetMapping(value = "/cs/createPosition")
255
    public String createPosition(HttpServletRequest request, @RequestParam(name = "offset", defaultValue = "0") int offset, @RequestParam(name = "limit", defaultValue = "10") int limit, Model model) {
256
        List<AuthUser> authUsers = authRepository.selectAllActiveUser();
257
        List<TicketCategory> ticketCategories = ticketCategoryRepository.selectAll();
258
        List<Region> regions = regionRepository.selectAll();
259
        model.addAttribute("escalationTypes", EscalationType.values());
260
        model.addAttribute("authUsers", authUsers);
261
        model.addAttribute("ticketCategories", ticketCategories);
262
        model.addAttribute("regions", regions);
24500 govind 263
 
31762 tejbeer 264
        List<Position> positions = positionRepository.selectAllPosition();
265
        LOGGER.info("positions" + positions);
30426 tejbeer 266
 
31762 tejbeer 267
        Map<Integer, AuthUser> authUserIdAndAuthUserMap = csService.getAuthUserIdAndAuthUserMapUsingPositions(positions);
268
        Map<Integer, TicketCategory> categoryIdAndCategoryMap = csService.getCategoryIdAndCategoryUsingPositions(positions);
269
        Map<Integer, Region> regionIdAndRegionMap = csService.getRegionIdAndRegionMap(positions);
25570 tejbeer 270
 
27410 tejbeer 271
//	    Map<Integer, List<CustomRetailer>> positionIdAndpartnerRegionMap = csService
272
//			.getpositionIdAndpartnerRegionMap(positions);
25570 tejbeer 273
 
27410 tejbeer 274
//	     Map<Integer, List<CustomRetailer>> addedpositionIdAndCustomRetailerMap = csService
275
//				.getPositionCustomRetailerMap(positions);
276
//		LOGGER.info("fofoIdAndCustomRetailerMap" + addedpositionIdAndCustomRetailerMap);
24500 govind 277
 
31762 tejbeer 278
        model.addAttribute("start", offset + 1);
30426 tejbeer 279
 
31762 tejbeer 280
        model.addAttribute("positions", positions);
281
        model.addAttribute("authUserIdAndAuthUserMap", authUserIdAndAuthUserMap);
282
        model.addAttribute("categoryIdAndCategoryMap", categoryIdAndCategoryMap);
283
        model.addAttribute("regionIdAndRegionMap", regionIdAndRegionMap);
284
        // model.addAttribute("positionIdAndCustomRetailerMap",
285
        // addedpositionIdAndCustomRetailerMap);
286
        // model.addAttribute("positionIdAndpartnerRegionMap",
27410 tejbeer 287
// positionIdAndpartnerRegionMap);
25570 tejbeer 288
 
31762 tejbeer 289
        return "create-position";
290
    }
24500 govind 291
 
31762 tejbeer 292
    @GetMapping(value = "/cs/position-paginated")
293
    public String positionPaginated(HttpServletRequest request, @RequestParam(name = "offset", defaultValue = "0") int offset, @RequestParam(name = "limit", defaultValue = "10") int limit, Model model) {
24500 govind 294
 
31762 tejbeer 295
        List<Position> positions = positionRepository.selectAll(offset, limit);
296
        Map<Integer, AuthUser> authUserIdAndAuthUserMap = csService.getAuthUserIdAndAuthUserMapUsingPositions(positions);
297
        Map<Integer, TicketCategory> categoryIdAndCategoryMap = csService.getCategoryIdAndCategoryUsingPositions(positions);
298
        Map<Integer, Region> regionIdAndRegionMap = csService.getRegionIdAndRegionMap(positions);
299
        /*
300
         * Map<Integer, List<CustomRetailer>> positionIdAndpartnerRegionMap = csService
301
         * .getpositionIdAndpartnerRegionMap(positions);
302
         * 
303
         * Map<Integer, List<CustomRetailer>> addedpositionIdAndCustomRetailerMap =
304
         * csService .getPositionCustomRetailerMap(positions);
305
         */
25570 tejbeer 306
 
31762 tejbeer 307
        model.addAttribute("positions", positions);
308
        model.addAttribute("authUserIdAndAuthUserMap", authUserIdAndAuthUserMap);
309
        model.addAttribute("categoryIdAndCategoryMap", categoryIdAndCategoryMap);
310
        model.addAttribute("regionIdAndRegionMap", regionIdAndRegionMap);
311
        // model.addAttribute("positionIdAndCustomRetailerMap",
312
        // addedpositionIdAndCustomRetailerMap);
313
        // model.addAttribute("positionIdAndpartnerRegionMap",
314
        // positionIdAndpartnerRegionMap);
25570 tejbeer 315
 
31762 tejbeer 316
        return "position-paginated";
317
    }
24417 govind 318
 
31762 tejbeer 319
    @PostMapping(value = "/cs/createPosition")
320
    public String createPosition(HttpServletRequest request, @RequestBody CreatePositionModel createPositionModel, Model model) throws Exception {
24417 govind 321
 
35570 amit 322
        LOGGER.info("partnerPosition" + createPositionModel.isTicketAssigned());
25570 tejbeer 323
 
35570 amit 324
        // Validate authUserId exists
325
        int authUserId = createPositionModel.getAuthUserId();
326
        if (authUserId <= 0 || authRepository.selectById(authUserId) == null) {
327
            throw new ProfitMandiBusinessException("Position", authUserId, "Invalid authUserId");
328
        }
329
 
330
        // Validate categoryId exists (0 means "All Categories")
331
        int categoryId = createPositionModel.getCategoryId();
332
        if (categoryId > 0 && ticketCategoryRepository.selectById(categoryId) == null) {
333
            throw new ProfitMandiBusinessException("Position", categoryId, "Invalid categoryId");
334
        }
335
 
336
        // Validate regionId exists (0 or 5 typically means "All Partners/Regions")
337
        int regionId = createPositionModel.getRegionId();
338
        if (regionId > 0 && regionId != 5 && regionRepository.selectById(regionId) == null) {
339
            throw new ProfitMandiBusinessException("Position", regionId, "Invalid regionId");
340
        }
341
 
35589 amit 342
        // Validate fofoIds exist (0 means "All Partners")
35570 amit 343
        List<Integer> fofoIds = createPositionModel.getFofoIds();
35590 amit 344
        if (fofoIds == null || fofoIds.isEmpty()) {
345
            throw new ProfitMandiBusinessException("Position", 0, "At least one partner must be specified");
346
        }
347
        boolean hasAllPartners = fofoIds.contains(0);
348
        if (hasAllPartners && fofoIds.size() > 1) {
349
            throw new ProfitMandiBusinessException("Position", 0, "Cannot mix 'All Partners' (0) with specific partner IDs");
350
        }
351
        if (!hasAllPartners) {
352
            Map<Integer, CustomRetailer> validRetailers = retailerService.getFofoRetailers(false);
353
            for (int fofoId : fofoIds) {
354
                if (!validRetailers.containsKey(fofoId)) {
355
                    throw new ProfitMandiBusinessException("Position", fofoId, "Invalid fofoId");
35570 amit 356
                }
357
            }
358
        }
359
 
360
        Position position = positionRepository.selectPosition(authUserId, categoryId, regionId, createPositionModel.getEscalationType());
31762 tejbeer 361
        if (position == null) {
362
            position = new Position();
35570 amit 363
            position.setAuthUserId(authUserId);
364
            position.setCategoryId(categoryId);
31762 tejbeer 365
            position.setEscalationType(createPositionModel.getEscalationType());
35570 amit 366
            position.setRegionId(regionId);
31762 tejbeer 367
            position.setCreateTimestamp(LocalDateTime.now());
368
            position.setTicketAssignee(createPositionModel.isTicketAssigned());
369
            positionRepository.persist(position);
25570 tejbeer 370
 
35570 amit 371
            if (fofoIds != null) {
372
                for (int fofoId : fofoIds) {
373
                    PartnerPosition partnerPosition = new PartnerPosition();
374
                    partnerPosition.setFofoId(fofoId);
375
                    partnerPosition.setRegionId(regionId);
376
                    partnerPosition.setPositionId(position.getId());
377
                    partnerPositionRepository.persist(partnerPosition);
378
                    LOGGER.info("partnerPosition" + partnerPosition);
379
                }
31762 tejbeer 380
            }
24417 govind 381
 
31762 tejbeer 382
            model.addAttribute("response1", mvcResponseSender.createResponseString(true));
383
        } else {
35570 amit 384
            throw new ProfitMandiBusinessException("Position", authUserId, "already exists!");
31762 tejbeer 385
        }
386
        return "response";
387
    }
25570 tejbeer 388
 
31762 tejbeer 389
    @PostMapping(value = "/cs/updatePartnerPosition")
390
    public String updatePartnerPosition(HttpServletRequest request, @RequestParam(name = "regionId") int regionId, @RequestBody List<Integer> selectedFofoIds, @RequestParam(name = "positionId") int positionId, Model model) throws Exception {
25570 tejbeer 391
 
35571 amit 392
        // Validate positionId exists
393
        Position position = positionRepository.selectById(positionId);
394
        if (position == null) {
395
            throw new ProfitMandiBusinessException("Position", positionId, "Position not found");
396
        }
397
 
398
        // Validate regionId exists (0 or 5 typically means "All Partners/Regions")
399
        if (regionId > 0 && regionId != 5 && regionRepository.selectById(regionId) == null) {
400
            throw new ProfitMandiBusinessException("Position", regionId, "Invalid regionId");
401
        }
402
 
35589 amit 403
        // Validate fofoIds exist (0 means "All Partners")
35590 amit 404
        if (selectedFofoIds == null || selectedFofoIds.isEmpty()) {
405
            throw new ProfitMandiBusinessException("Position", 0, "At least one partner must be specified");
406
        }
407
        boolean hasAllPartners = selectedFofoIds.contains(0);
408
        if (hasAllPartners && selectedFofoIds.size() > 1) {
409
            throw new ProfitMandiBusinessException("Position", 0, "Cannot mix 'All Partners' (0) with specific partner IDs");
410
        }
411
        if (!hasAllPartners) {
35607 amit 412
            // Use direct database query instead of cached retailerService to include newly added partners
413
            Set<Integer> validFofoIds = fofoStoreRepository.selectAll().stream()
414
                    .map(fs -> fs.getId())
415
                    .collect(Collectors.toSet());
35590 amit 416
            for (int fofoId : selectedFofoIds) {
35607 amit 417
                if (!validFofoIds.contains(fofoId)) {
35590 amit 418
                    throw new ProfitMandiBusinessException("Position", fofoId, "Invalid fofoId");
35571 amit 419
                }
420
            }
421
        }
422
 
32493 amit.gupta 423
        partnerPositionRepository.delete(positionId);
35571 amit 424
        if (selectedFofoIds != null) {
425
            for (int fofoId : selectedFofoIds) {
426
                PartnerPosition partnerPosition = new PartnerPosition();
427
                partnerPosition.setFofoId(fofoId);
428
                partnerPosition.setRegionId(regionId);
429
                partnerPosition.setPositionId(positionId);
430
                partnerPositionRepository.persist(partnerPosition);
431
            }
31762 tejbeer 432
        }
25570 tejbeer 433
 
31762 tejbeer 434
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
25570 tejbeer 435
 
31762 tejbeer 436
        return "response";
437
    }
24417 govind 438
 
31762 tejbeer 439
    @GetMapping(value = "/cs/createTicket")
440
    public String createTicket(HttpServletRequest request, Model model) throws ProfitMandiBusinessException {
441
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
35594 amit 442
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
443
        boolean isCrmUser = false;
444
 
31762 tejbeer 445
        List<TicketCategory> ticketCategories = csService.getAllTicketCategotyFromSubCategory();
35594 amit 446
 
447
        if (isAdmin) {
448
            AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
449
            isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
450
 
451
            // CRM users should not see Sales and RBM categories
452
            if (isCrmUser) {
453
                ticketCategories = ticketCategories.stream()
454
                        .filter(tc -> tc.getId() != ProfitMandiConstants.TICKET_CATEGORY_SALES
455
                                && tc.getId() != ProfitMandiConstants.TICKET_CATEGORY_RBM)
456
                        .collect(Collectors.toList());
457
            }
35626 amit 458
        } else {
459
            // For partners/customers: hide categories where ALL subcategories are invisible
460
            ticketCategories = ticketCategories.stream()
461
                    .filter(category -> {
462
                        List<TicketSubCategory> subs = ticketSubCategoryRepository.selectAll(category.getId());
463
                        // Keep category if at least one subcategory is visible
464
                        return subs.stream().anyMatch(TicketSubCategory::isVisibility);
465
                    })
466
                    .collect(Collectors.toList());
35594 amit 467
        }
468
 
469
        model.addAttribute("roleType", isAdmin);
31762 tejbeer 470
        model.addAttribute("ticketCategories", ticketCategories);
35594 amit 471
        model.addAttribute("isCrmUser", isCrmUser);
31762 tejbeer 472
        return "create-ticket";
473
    }
24417 govind 474
 
31762 tejbeer 475
    @GetMapping(value = "/cs/getSubCategoriesByCategoryId")
35594 amit 476
    public String getSubCategoriesByCategoryId(HttpServletRequest request, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, Model model) throws ProfitMandiBusinessException {
477
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
478
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
479
        boolean isCrmUser = false;
480
 
481
        if (isAdmin) {
482
            AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
483
            isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
484
        }
485
 
31830 amit.gupta 486
        List<TicketSubCategory> ticketSubCategories = ticketSubCategoryRepository.selectAllVisible(categoryId);
35594 amit 487
 
35617 amit 488
        // CRM users should not see invisible subcategories
35594 amit 489
        if (isCrmUser) {
490
            ticketSubCategories = ticketSubCategories.stream()
35617 amit 491
                    .filter(TicketSubCategory::isVisibility)
35594 amit 492
                    .collect(Collectors.toList());
493
        }
494
 
31762 tejbeer 495
        LOGGER.info(ticketSubCategories);
496
        model.addAttribute("ticketSubCategories", ticketSubCategories);
497
        return "ticket-sub-categories";
498
    }
24417 govind 499
 
24791 govind 500
 
31762 tejbeer 501
    @GetMapping(value = "/cs/getEscalationTypeByCategoryId")
502
    public String getEscalationTypeByCategoryId(HttpServletRequest request, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, @RequestParam(name = "authId", defaultValue = "0") int authId, Model model) {
503
        List<Position> positions = positionRepository.selectPositionbyCategoryIdAndAuthId(categoryId, authId);
504
        List<EscalationType> escalationTypes = new ArrayList<>();
24791 govind 505
 
31762 tejbeer 506
        if (!positions.isEmpty()) {
507
            escalationTypes = positions.stream().map(x -> x.getEscalationType()).distinct().collect(Collectors.toList());
508
        }
24620 govind 509
 
31762 tejbeer 510
        LOGGER.info("escalationTypes {}", escalationTypes);
24500 govind 511
 
31762 tejbeer 512
        model.addAttribute("escalationTypes", escalationTypes);
513
        return "ticket-escalationtype";
514
    }
24500 govind 515
 
24824 govind 516
 
31762 tejbeer 517
    @GetMapping(value = "/cs/getCategoriesByAuthId")
518
    public String getCategoriesByAuthId(HttpServletRequest request, @RequestParam(name = "authId", defaultValue = "0") int authId, Model model) {
24824 govind 519
 
24787 govind 520
 
31762 tejbeer 521
        List<Position> positions = positionRepository.selectPositionByAuthId(authId);
24791 govind 522
 
31762 tejbeer 523
        LOGGER.info("positions {}", positions);
24417 govind 524
 
31762 tejbeer 525
        List<TicketCategory> ticketCategories = new ArrayList<TicketCategory>();
24417 govind 526
 
31762 tejbeer 527
        if (!positions.isEmpty()) {
24417 govind 528
 
31762 tejbeer 529
            List<Integer> categoryIds = positions.stream().map(x -> x.getCategoryId()).collect(Collectors.toList());
530
            ticketCategories = ticketCategoryRepository.selectAll(categoryIds);
531
        }
532
        LOGGER.info("ticketCategories {}", ticketCategories);
533
        model.addAttribute("ticketCategories", ticketCategories);
534
        return "ticket-categories";
535
    }
27270 tejbeer 536
 
31762 tejbeer 537
    @PostMapping(value = "/cs/createTicket")
538
    public String createTicket(HttpServletRequest request, @RequestParam(name = "categoryId") int categoryId, @RequestParam(name = "subCategoryId") int subCategoryId, @RequestParam(name = "message") String message, Model model) throws Exception {
539
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
540
        List<Ticket> tickets = ticketRepository.selectAllResolvedMarkedTicketByCreator(loginDetails.getFofoId());
541
        if (tickets.size() > 3 || tickets.size() == 3) {
542
            model.addAttribute("response1", mvcResponseSender.createResponseString(false));
543
        } else {
544
            csService.createTicket(loginDetails.getFofoId(), categoryId, subCategoryId, message);
545
            model.addAttribute("response1", mvcResponseSender.createResponseString(true));
546
        }
547
        return "response";
548
    }
27270 tejbeer 549
 
31762 tejbeer 550
    @GetMapping(value = "/cs/myticket")
35626 amit 551
    public String getTicket(HttpServletRequest request,
552
                            @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
553
                            @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
554
                            @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
555
                            @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
556
                            @RequestParam(name = "page", defaultValue = "0") int page,
557
                            @RequestParam(name = "pageSize", defaultValue = "25") int pageSize,
558
                            @RequestParam(name = "search", required = false) String searchText,
559
                            Model model) throws ProfitMandiBusinessException {
560
        populateMyTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, page, pageSize, searchText, model);
561
        return "ticket";
562
    }
563
 
564
    @GetMapping(value = "/cs/myticket-content")
565
    public String getTicketContent(HttpServletRequest request,
566
                                   @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
567
                                   @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
568
                                   @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
569
                                   @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
570
                                   @RequestParam(name = "page", defaultValue = "0") int page,
571
                                   @RequestParam(name = "pageSize", defaultValue = "25") int pageSize,
572
                                   @RequestParam(name = "search", required = false) String searchText,
573
                                   Model model) throws ProfitMandiBusinessException {
574
        populateMyTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, page, pageSize, searchText, model);
575
        return "ticket-content";
576
    }
577
 
578
    private void populateMyTicketModel(HttpServletRequest request, SortOrder sortOrder, TicketStatus ticketStatus,
579
                                       TicketSearchType ticketSearchType, int searchTerm, int page, int pageSize,
580
                                       String searchText, Model model) throws ProfitMandiBusinessException {
31762 tejbeer 581
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
582
        List<Ticket> tickets = null;
583
        List<TicketAssigned> ticketAssigneds = null;
35626 amit 584
        long totalRecords = 0;
31762 tejbeer 585
        Map<Integer, AuthUser> authUserIdAndAuthUserMap = null;
35569 amit 586
        boolean isAdmin = roleManager.isAdmin(new HashSet<>(loginDetails.getRoleIds()));
587
        AuthUser currentAuthUser = isAdmin ? authRepository.selectByEmailOrMobile(loginDetails.getEmailId()) : null;
588
 
35592 amit 589
        boolean isCrmUser = isAdmin && currentAuthUser != null && positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
35603 amit 590
        boolean isSalesUser = isAdmin && currentAuthUser != null && positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_SALES);
591
        boolean isRbmUser = isAdmin && currentAuthUser != null && positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_RBM);
35592 amit 592
 
35626 amit 593
        List<Integer> userCategoryIds = isAdmin && currentAuthUser != null
594
                ? positionRepository.selectCategoryIdsByAuthUserId(currentAuthUser.getId())
595
                : Collections.emptyList();
596
 
597
        int offset = page * pageSize;
598
 
35569 amit 599
        if (isAdmin) {
600
            int authUserId = currentAuthUser.getId();
35592 amit 601
 
602
            if (isCrmUser) {
35626 amit 603
                Optional<Boolean> resolvedOpt = ticketStatus.equals(TicketStatus.RESOLVED) ? Optional.empty() : Optional.of(TicketStatus.CLOSED.equals(ticketStatus));
604
                tickets = ticketRepository.selectAllTicketsPaginated(
605
                        resolvedOpt, sortOrder, ticketSearchType, searchTerm, searchText,
606
                        userCategoryIds, offset, pageSize);
607
                totalRecords = ticketRepository.selectAllTicketsPaginatedCount(
608
                        resolvedOpt, ticketSearchType, searchTerm, searchText, userCategoryIds);
31762 tejbeer 609
            } else {
35626 amit 610
                Optional<Boolean> resolvedOpt = ticketStatus.equals(TicketStatus.RESOLVED) ? Optional.empty() : Optional.of(TicketStatus.CLOSED.equals(ticketStatus));
611
                tickets = ticketRepository.selectAllByAssigneePaginated(
612
                        authUserId, resolvedOpt, sortOrder, ticketSearchType, searchTerm, searchText,
613
                        offset, pageSize);
614
                totalRecords = ticketRepository.selectAllByAssigneePaginatedCount(
615
                        authUserId, resolvedOpt, ticketSearchType, searchTerm, searchText);
31762 tejbeer 616
            }
35626 amit 617
 
31762 tejbeer 618
            if (tickets.size() > 0) {
619
                ticketAssigneds = ticketAssignedRepository.selectByTicketIds(tickets.stream().map(x -> x.getId()).collect(Collectors.toList()));
620
                authUserIdAndAuthUserMap = csService.getAuthUserIdAndAuthUserMap(ticketAssigneds);
621
                Map<Integer, CustomRetailer> fofoIdsAndCustomRetailer = csService.getPartnerByFofoIds(tickets);
622
                model.addAttribute("fofoIdsAndCustomRetailer", fofoIdsAndCustomRetailer);
623
            }
24500 govind 624
 
31762 tejbeer 625
        } else {
626
            tickets = ticketRepository.selectAllByCreator(loginDetails.getFofoId(), Optional.of(TicketStatus.OPENED.equals(ticketStatus)), sortOrder);
35626 amit 627
            totalRecords = ticketRepository.selectAllCountByCreator(loginDetails.getFofoId(), Optional.of(TicketStatus.OPENED.equals(ticketStatus)));
31762 tejbeer 628
        }
629
        authUserIdAndAuthUserMap = csService.getTicketIdAndAuthUserMapUsingTickets(tickets);
35571 amit 630
        if (authUserIdAndAuthUserMap == null) {
631
            authUserIdAndAuthUserMap = new HashMap<>();
632
        }
24620 govind 633
 
36040 ranu 634
        Map<Integer, List<AuthUser>> authUserListMap = csService.getAssignedAuthList(tickets);
635
        if (authUserListMap == null) {
636
            authUserListMap = new HashMap<>();
637
        }
638
 
35626 amit 639
        int totalPages = (int) Math.ceil((double) totalRecords / pageSize);
640
        if (totalPages == 0) totalPages = 1;
641
        int startRecord = totalRecords > 0 ? offset + 1 : 0;
642
        int endRecord = (int) Math.min(offset + pageSize, totalRecords);
643
 
644
        model.addAttribute("size", totalRecords);
645
        model.addAttribute("totalRecords", totalRecords);
646
        model.addAttribute("currentPage", page);
647
        model.addAttribute("pageSize", pageSize);
648
        model.addAttribute("totalPages", totalPages);
649
        model.addAttribute("startRecord", startRecord);
650
        model.addAttribute("endRecord", endRecord);
651
        model.addAttribute("searchText", searchText != null ? searchText : "");
652
        model.addAttribute("currentPageDisplay", page + 1);
653
        model.addAttribute("prevPage", page - 1);
654
        model.addAttribute("nextPage", page + 1);
655
        model.addAttribute("lastPage", totalPages - 1);
656
 
35569 amit 657
        model.addAttribute("roleType", isAdmin);
658
        model.addAttribute("isCrmUser", isCrmUser);
35603 amit 659
        model.addAttribute("isSalesUser", isSalesUser);
660
        model.addAttribute("salesCategoryId", ProfitMandiConstants.TICKET_CATEGORY_SALES);
661
        model.addAttribute("isRbmUser", isRbmUser);
662
        model.addAttribute("rbmCategoryId", ProfitMandiConstants.TICKET_CATEGORY_RBM);
35626 amit 663
        model.addAttribute("userCategoryIds", userCategoryIds);
24500 govind 664
 
31762 tejbeer 665
        List<Integer> subCategoryIds = tickets.stream().map(x -> x.getSubCategoryId()).collect(Collectors.toList());
666
        Map<Integer, TicketSubCategory> subCategoryIdAndSubCategoryMap = csService.getSubCategoryIdAndSubCategoryMap(subCategoryIds);
35571 amit 667
        if (subCategoryIdAndSubCategoryMap == null) {
668
            subCategoryIdAndSubCategoryMap = new HashMap<>();
669
        }
24747 govind 670
 
31762 tejbeer 671
        Map<Integer, TicketCategory> subCategoryIdAndCategoryMap = csService.getSubCategoryIdAndCategoryMap(subCategoryIds);
35571 amit 672
        if (subCategoryIdAndCategoryMap == null) {
673
            subCategoryIdAndCategoryMap = new HashMap<>();
674
        }
27318 amit.gupta 675
 
31762 tejbeer 676
        List<Integer> ticketIds = tickets.stream().map(x -> x.getId()).collect(Collectors.toList());
24500 govind 677
 
35569 amit 678
        Map<Integer, List<Activity>> activityMap = new HashMap<>();
33864 ranu 679
        Map<Integer, List<Activity>> activityMapWithActivityId = new HashMap<>();
35569 amit 680
        Map<Integer, AuthUser> authUserMap = new HashMap<>();
24500 govind 681
 
31762 tejbeer 682
        if (!ticketIds.isEmpty()) {
35569 amit 683
            List<Activity> allActivities = activityRepository.selectAll(ticketIds);
684
            activityMap = allActivities.stream().collect(Collectors.groupingBy(Activity::getTicketId));
685
            activityMapWithActivityId = allActivities.stream().collect(Collectors.groupingBy(Activity::getId));
24787 govind 686
 
35569 amit 687
            Set<Integer> activityCreatorIds = allActivities.stream()
688
                    .map(Activity::getCreatedBy)
689
                    .filter(id -> id > 0)
690
                    .collect(Collectors.toSet());
691
            if (!activityCreatorIds.isEmpty()) {
692
                authUserMap = authRepository.selectByIds(new ArrayList<>(activityCreatorIds))
693
                        .stream().collect(Collectors.toMap(AuthUser::getId, x -> x));
694
            }
31762 tejbeer 695
        }
24787 govind 696
 
35626 amit 697
        int currentUserId;
698
        UserType currentUserType;
699
        if (isAdmin) {
700
            currentUserId = currentAuthUser.getId();
701
            currentUserType = UserType.AUTH_USER;
702
        } else {
703
            currentUserId = loginDetails.getFofoId();
704
            currentUserType = UserType.PARTNER;
705
        }
706
        Map<Integer, Boolean> unreadMap = csService.getUnreadStatusForTickets(tickets, currentUserId, currentUserType);
707
        Map<Integer, Activity> lastActivityMap = csService.getLastActivitiesForTickets(ticketIds);
35640 amit 708
        Map<Integer, Activity> lastMessageMap = csService.getLastMessageActivitiesForTickets(ticketIds);
35626 amit 709
        model.addAttribute("unreadMap", unreadMap);
710
        model.addAttribute("lastActivityMap", lastActivityMap);
35640 amit 711
        model.addAttribute("lastMessageMap", lastMessageMap);
35626 amit 712
 
31762 tejbeer 713
        model.addAttribute("tickets", tickets);
714
        model.addAttribute("resolved", ActivityType.RESOLVED);
715
        model.addAttribute("resolved-accepted", ActivityType.RESOLVED_ACCEPTED);
716
        model.addAttribute("resolved-rejected", ActivityType.RESOLVED_REJECTED);
717
        model.addAttribute("authUserIdAndAuthUserMap", authUserIdAndAuthUserMap);
36040 ranu 718
        model.addAttribute("authUserListMap", authUserListMap);
31762 tejbeer 719
        model.addAttribute("subCategoryIdAndSubCategoryMap", subCategoryIdAndSubCategoryMap);
27318 amit.gupta 720
 
31762 tejbeer 721
        model.addAttribute("subCategoryIdAndCategoryMap", subCategoryIdAndCategoryMap);
722
        model.addAttribute("activityMap", activityMap);
33864 ranu 723
        model.addAttribute("authUserMap", authUserMap);
724
        model.addAttribute("activityMapWithActivityId", activityMapWithActivityId);
24500 govind 725
 
31762 tejbeer 726
        model.addAttribute("ticketStatusValues", TicketStatus.values());
727
        model.addAttribute("orderByValues", SortOrder.values());
728
        model.addAttribute("selectedticketStatus", ticketStatus);
729
        model.addAttribute("selectedorderby", sortOrder);
730
        model.addAttribute("ticketSearchTypes", TicketSearchType.values());
731
        model.addAttribute("ticketSearchType", ticketSearchType);
732
        model.addAttribute("searchTerm", searchTerm);
733
    }
24500 govind 734
 
27124 amit.gupta 735
 
31762 tejbeer 736
    @GetMapping(value = "/cs/getActivities")
737
    public String getActivity(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, Model model) throws Exception {
35592 amit 738
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
739
        Ticket ticket = ticketRepository.selectById(ticketId);
740
 
741
        if (ticket == null) {
742
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
743
        }
744
 
745
        // Authorization check: verify user has access to this ticket
746
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
747
        if (!isAdmin) {
748
            // Partners can only view their own tickets
749
            if (ticket.getFofoId() != loginDetails.getFofoId()) {
750
                throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to view this ticket");
751
            }
752
        } else {
753
            // Admins must be assigned to the ticket OR be CRM user OR be in the escalation chain
754
            AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
755
            boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
756
 
757
            if (!isCrmUser) {
758
                // Check if user is assigned or in escalation chain
759
                List<TicketAssigned> assignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
760
                boolean isAssigned = assignments.stream().anyMatch(ta -> ta.getAssineeId() == currentAuthUser.getId() || ta.getManagerId() == currentAuthUser.getId());
761
                boolean isInEscalationChain = ticket.getL1AuthUser() == currentAuthUser.getId() ||
762
                        ticket.getL2AuthUser() == currentAuthUser.getId() ||
763
                        ticket.getL3AuthUser() == currentAuthUser.getId() ||
764
                        ticket.getL4AuthUser() == currentAuthUser.getId() ||
765
                        ticket.getL5AuthUser() == currentAuthUser.getId();
766
 
767
                if (!isAssigned && !isInEscalationChain) {
768
                    throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to view this ticket");
769
                }
770
            }
771
        }
772
 
31762 tejbeer 773
        List<Activity> allactivities = activityRepository.selectAll(ticketId);
35570 amit 774
 
775
        // Batch fetch all documents for attachments (fix N+1 query)
776
        List<Integer> documentIds = allactivities.stream()
777
                .flatMap(a -> a.getActivityAttachment().stream())
778
                .map(ActivityAttachment::getDocumentId)
779
                .distinct()
780
                .collect(Collectors.toList());
781
 
782
        Map<Integer, Document> documentMap = Collections.emptyMap();
783
        if (!documentIds.isEmpty()) {
784
            documentMap = documentRepository.selectByIds(documentIds).stream()
785
                    .collect(Collectors.toMap(Document::getId, d -> d));
786
        }
787
 
788
        // Set document names transiently (do not persist during GET)
789
        for (Activity activity : allactivities) {
790
            for (ActivityAttachment attachment : activity.getActivityAttachment()) {
791
                Document document = documentMap.get(attachment.getDocumentId());
792
                if (document != null) {
793
                    attachment.setDocumentName(document.getDisplayName());
794
                }
31762 tejbeer 795
            }
796
        }
797
        List<Activity> activities = null;
35592 amit 798
        if (isAdmin) {
31762 tejbeer 799
            Set<Integer> authUserIds = allactivities.stream().map(x -> x.getCreatedBy()).collect(Collectors.toSet());
36040 ranu 800
 
801
            // Fetch current assignees for this ticket to show in ASSIGNED/ESCALATED activities
802
            List<TicketAssigned> ticketAssignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
803
            Set<Integer> assigneeIds = ticketAssignments.stream()
804
                    .map(TicketAssigned::getAssineeId)
805
                    .collect(Collectors.toSet());
806
            authUserIds.addAll(assigneeIds);
807
 
35395 amit 808
            List<AuthUser> users = authRepository.selectByIds(new ArrayList<>(authUserIds));
31762 tejbeer 809
            Map<Integer, String> authUserNameMap = users.stream().collect(Collectors.toMap(AuthUser::getId, x -> x.getFirstName() + " " + x.getLastName()));
36040 ranu 810
 
811
            // Get current assignee name (most recent assignment)
812
            String currentAssigneeName = null;
813
            if (!ticketAssignments.isEmpty()) {
814
                int latestAssigneeId = ticketAssignments.get(ticketAssignments.size() - 1).getAssineeId();
815
                currentAssigneeName = authUserNameMap.get(latestAssigneeId);
816
            }
817
            final String assigneeName = currentAssigneeName;
818
 
819
            allactivities.stream().forEach(x -> {
820
                x.setName(authUserNameMap.get(x.getCreatedBy()));
821
                // For ASSIGNED/ESCALATED activities, use name field to show assignee
822
                ActivityType actType = x.getType();
823
                if (assigneeName != null && (actType == ActivityType.ASSIGNED || actType == ActivityType.ESCALATED || actType == ActivityType.ESCALATED_FINAL)) {
824
                    x.setName(assigneeName);
825
                }
826
            });
31762 tejbeer 827
            activities = allactivities;
828
        } else {
829
            activities = allactivities.stream().filter(x -> ActivityType.PARTNER_ACTIVITIES.contains(x.getType())).collect(Collectors.toList());
830
        }
831
        if (activities == null) {
832
            throw new ProfitMandiBusinessException("Activity", ticketId, "No Activity Found");
833
        }
834
        model.addAttribute("response1", mvcResponseSender.createResponseString(activities));
835
        return "response";
24500 govind 836
 
31762 tejbeer 837
    }
24620 govind 838
 
31762 tejbeer 839
    @PostMapping(value = "/cs/createActivity")
840
    public String createActivity(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, @RequestParam(name = "assigneeId", defaultValue = "0") int assigneeId, @RequestParam(name = "internal", defaultValue = "true") boolean internal, @RequestParam(name = "message", defaultValue = "") String message, @RequestBody List<Integer> documentIds,
24620 govind 841
 
31762 tejbeer 842
                                 Model model) throws Exception {
24620 govind 843
 
31762 tejbeer 844
        LOGGER.info("documentIds" + documentIds);
845
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
846
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 847
 
848
        if (ticket == null) {
849
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
850
        }
851
 
852
        // Authorization check: verify user has access to add activity to this ticket
853
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
854
        if (!isAdmin) {
855
            // Partners can only add activity to their own tickets
856
            if (ticket.getFofoId() != loginDetails.getFofoId()) {
857
                throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to add activity to this ticket");
858
            }
859
        } else {
860
            // Admins must be assigned to the ticket OR be CRM user OR be in the escalation chain
861
            AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
862
            boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
863
 
864
            if (!isCrmUser) {
865
                List<TicketAssigned> assignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
866
                boolean isAssigned = assignments.stream().anyMatch(ta -> ta.getAssineeId() == currentAuthUser.getId() || ta.getManagerId() == currentAuthUser.getId());
867
                boolean isInEscalationChain = ticket.getL1AuthUser() == currentAuthUser.getId() ||
868
                        ticket.getL2AuthUser() == currentAuthUser.getId() ||
869
                        ticket.getL3AuthUser() == currentAuthUser.getId() ||
870
                        ticket.getL4AuthUser() == currentAuthUser.getId() ||
871
                        ticket.getL5AuthUser() == currentAuthUser.getId();
872
 
873
                if (!isAssigned && !isInEscalationChain) {
874
                    throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to add activity to this ticket");
875
                }
876
            }
877
        }
31762 tejbeer 878
        List<TicketAssigned> ticketAssignedList = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
879
        List<Integer> authUserIds = ticketAssignedList.stream().map(x -> x.getAssineeId()).collect(Collectors.toList());
880
        authUserIds.add(ticketAssignedList.get(ticketAssignedList.size() - 1).getManagerId());
35395 amit 881
        Map<Integer, AuthUser> authUsersMap = authRepository.selectByIds(authUserIds).stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
31762 tejbeer 882
        if (ticket.getCloseTimestamp() == null) {
883
            Activity activity = new Activity();
884
            activity.setCreatedBy(0);
885
            activity.setCreateTimestamp(LocalDateTime.now());
886
            String subject = null;
887
            String mailMessage = null;
888
            activity.setMessage(message);
889
            if (!roleManager.isAdmin(new HashSet<>(loginDetails.getRoleIds()))) {
890
                CustomRetailer customRetailer = retailerService.getFofoRetailers(true).get(loginDetails.getFofoId());
891
                activity.setType(ActivityType.COMMUNICATION_IN);
892
                subject = String.format("Ticket Update #%s by franchisee %s", ticket.getId(), customRetailer.getBusinessName() + "(" + customRetailer.getCode() + ")");
893
                mailMessage = String.format("Franchisee message - %s", message);
894
            } else {
895
                AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
896
                activity.setCreatedBy(authUser.getId());
897
                authUsersMap.remove(authUser.getId());
898
                subject = String.format("Ticket Update #%s by %s", ticket.getId(), authUser.getName());
899
                mailMessage = String.format("%s's message - %s", authUser.getFirstName(), message);
35569 amit 900
                // Only CRM users can send external communications
901
                boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
902
                if (internal || !isCrmUser) {
31762 tejbeer 903
                    activity.setType(ActivityType.COMMUNICATION_INTERNAL);
904
                } else {
905
                    String updatedBy = "SD Team";
31854 amit.gupta 906
                    CustomRetailer customRetailer = retailerService.getFofoRetailers(false).get(ticket.getFofoId());
31762 tejbeer 907
                    subject = String.format("Ticket Update #%s by %s", ticket.getId(), updatedBy);
908
                    String partnerMessage = String.format("%s's message - %s", updatedBy, message);
909
                    this.activityRelatedMail(customRetailer.getEmail(), null, "subject", partnerMessage);
910
                    activity.setType(ActivityType.COMMUNICATION_OUT);
911
                }
912
            }
913
            activityRepository.persist(activity);
914
 
915
            for (Integer documentId : documentIds) {
916
                ActivityAttachment activityAttachment = new ActivityAttachment();
917
                activityAttachment.setActivityId(activity.getId());
918
                activityAttachment.setDocumentId(documentId);
919
                activityAttachmentRepository.persist(activityAttachment);
920
            }
921
 
922
            csService.addActivity(ticket, activity);
923
            AuthUser authUser = authUsersMap.remove(authUserIds.get(0));
924
            if (authUser == null) {
925
                authUser = authUsersMap.remove(authUserIds.get(1));
926
            }
927
            model.addAttribute("response1", mvcResponseSender.createResponseString(authUser));
928
            String[] cc = authUsersMap.entrySet().stream().map(x -> x.getValue().getEmailId()).toArray(String[]::new);
929
            this.activityRelatedMail(authUser.getEmailId(), cc, subject, mailMessage);
930
        } else {
931
            throw new ProfitMandiBusinessException("Ticket", ticket.getId(), "Already closed ticket");
932
        }
933
        return "response";
934
    }
935
 
936
    private void activityRelatedMail(String to, String[] cc, String subject, String message) throws ProfitMandiBusinessException {
937
        try {
35957 amit 938
            mailOutboxService.queueMail(to, cc, subject, message, "CsController.activityRelatedMail");
31762 tejbeer 939
        } catch (Exception e) {
940
            throw new ProfitMandiBusinessException("Ticket Activity", to, "Could not send ticket activity mail");
941
        }
942
    }
943
 
944
    @PostMapping(value = "/cs/closeTicket")
945
    public String closeTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, @RequestParam(name = "happyCode") String happyCode, Model model) throws Exception {
35592 amit 946
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
31762 tejbeer 947
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 948
 
949
        if (ticket == null) {
950
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
951
        }
952
 
953
        // Authorization check: only the ticket owner (partner) can close the ticket
954
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
955
        if (isAdmin) {
956
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Only partners can close tickets using happy code");
957
        }
958
 
959
        // Verify the partner owns this ticket
960
        if (ticket.getFofoId() != loginDetails.getFofoId()) {
961
            throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to close this ticket");
962
        }
963
 
31762 tejbeer 964
        if (ticket.getHappyCode().equals(happyCode)) {
965
            ticket.setCloseTimestamp(LocalDateTime.now());
966
            ticketRepository.persist(ticket);
967
            model.addAttribute("response1", mvcResponseSender.createResponseString(true));
968
        } else {
969
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Happy Code can't match");
970
        }
971
        return "response";
972
    }
973
 
34913 ranu 974
    @GetMapping(value = "/cs/myPartyTicketTicket")
975
    public String getMyPartyTicketTicket(HttpServletRequest request, @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder, @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus, @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType, @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm, Model model) throws ProfitMandiBusinessException {
35626 amit 976
        populateMyPartnerTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, model);
977
        return "my-partner-tickets";
978
    }
979
 
980
    @GetMapping(value = "/cs/myPartyTicket-content")
981
    public String getMyPartyTicketContent(HttpServletRequest request,
982
                                          @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
983
                                          @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
984
                                          @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
985
                                          @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
986
                                          Model model) throws ProfitMandiBusinessException {
987
        populateMyPartnerTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, model);
988
        return "my-partner-tickets-content";
989
    }
990
 
991
    private void populateMyPartnerTicketModel(HttpServletRequest request, SortOrder sortOrder, TicketStatus ticketStatus,
992
                                              TicketSearchType ticketSearchType, int searchTerm, Model model) throws ProfitMandiBusinessException {
34913 ranu 993
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
994
        AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
995
        List<Ticket> tickets = new ArrayList<>();
996
        Map<String, Set<Integer>> storeGuyMap = csService.getAuthUserPartnerIdMapping();
997
        Set<Integer> fofoIds = storeGuyMap.get(authUser.getEmailId());
998
 
999
        Map<Integer, List<AuthUser>> authUserListMap = null;
1000
        if (fofoIds != null && !fofoIds.isEmpty()) {
35571 amit 1001
            List<Ticket> allPartnerTickets = ticketRepository.selectAllOpenTickets(new ArrayList<>(fofoIds));
1002
            if (allPartnerTickets != null && !allPartnerTickets.isEmpty()) {
1003
                tickets = allPartnerTickets.stream()
1004
                        .filter(ticket -> ticket.getLastActivity() == null ||
1005
                                ticket.getLastActivity() != ActivityType.RESOLVED)
1006
                        .collect(Collectors.toList());
34913 ranu 1007
            }
1008
        }
1009
 
1010
        authUserListMap = csService.getAssignedAuthList(tickets);
1011
 
1012
        if (tickets.size() > 0) {
1013
            Map<Integer, CustomRetailer> fofoIdsAndCustomRetailer = csService.getPartnerByFofoIds(tickets);
1014
            model.addAttribute("fofoIdsAndCustomRetailer", fofoIdsAndCustomRetailer);
1015
        }
1016
 
35626 amit 1017
        // Pagination for partner tickets (in-memory since selectAllOpenTickets doesn't support DB-level pagination)
1018
        long totalRecords = tickets.size();
1019
        int totalPages = totalRecords > 0 ? (int) Math.ceil((double) totalRecords / 25) : 1;
1020
        model.addAttribute("totalRecords", totalRecords);
1021
        model.addAttribute("currentPage", 0);
1022
        model.addAttribute("pageSize", 25);
1023
        model.addAttribute("totalPages", totalPages);
1024
        model.addAttribute("startRecord", totalRecords > 0 ? 1 : 0);
1025
        model.addAttribute("endRecord", totalRecords);
1026
        model.addAttribute("currentPageDisplay", 1);
1027
        model.addAttribute("prevPage", 0);
1028
        model.addAttribute("nextPage", 1);
1029
        model.addAttribute("lastPage", totalPages - 1);
1030
 
34913 ranu 1031
        model.addAttribute("size", tickets.size());
1032
        model.addAttribute("tickets", tickets);
1033
 
1034
        List<Integer> subCategoryIds = tickets.stream().map(x -> x.getSubCategoryId()).collect(Collectors.toList());
1035
        Map<Integer, TicketSubCategory> subCategoryIdAndSubCategoryMap = csService.getSubCategoryIdAndSubCategoryMap(subCategoryIds);
35571 amit 1036
        if (subCategoryIdAndSubCategoryMap == null) {
1037
            subCategoryIdAndSubCategoryMap = new HashMap<>();
1038
        }
34913 ranu 1039
 
1040
        Map<Integer, TicketCategory> subCategoryIdAndCategoryMap = csService.getSubCategoryIdAndCategoryMap(subCategoryIds);
35571 amit 1041
        if (subCategoryIdAndCategoryMap == null) {
1042
            subCategoryIdAndCategoryMap = new HashMap<>();
1043
        }
34913 ranu 1044
 
1045
        List<Integer> ticketIds = tickets.stream().map(x -> x.getId()).collect(Collectors.toList());
1046
        Map<Integer, List<Activity>> activityMap = new HashMap<>();
1047
        Map<Integer, List<Activity>> activityMapWithActivityId = new HashMap<>();
35569 amit 1048
        Map<Integer, AuthUser> authUserMap = new HashMap<>();
34913 ranu 1049
 
1050
        if (!ticketIds.isEmpty()) {
35569 amit 1051
            List<Activity> allActivities = activityRepository.selectAll(ticketIds);
1052
            activityMap = allActivities.stream().collect(Collectors.groupingBy(Activity::getTicketId));
1053
            activityMapWithActivityId = allActivities.stream().collect(Collectors.groupingBy(Activity::getId));
34913 ranu 1054
 
35569 amit 1055
            Set<Integer> activityCreatorIds = allActivities.stream()
1056
                    .map(Activity::getCreatedBy)
1057
                    .filter(id -> id > 0)
1058
                    .collect(Collectors.toSet());
1059
            if (!activityCreatorIds.isEmpty()) {
1060
                authUserMap = authRepository.selectByIds(new ArrayList<>(activityCreatorIds))
1061
                        .stream().collect(Collectors.toMap(AuthUser::getId, x -> x));
1062
            }
34913 ranu 1063
        }
1064
 
35626 amit 1065
        Map<Integer, Boolean> unreadMap = csService.getUnreadStatusForTickets(tickets, authUser.getId(), UserType.AUTH_USER);
35640 amit 1066
        List<Integer> partnerTicketIds = tickets.stream().map(Ticket::getId).collect(Collectors.toList());
1067
        Map<Integer, Activity> lastActivityMap = csService.getLastActivitiesForTickets(partnerTicketIds);
1068
        Map<Integer, Activity> lastMessageMap = csService.getLastMessageActivitiesForTickets(partnerTicketIds);
35626 amit 1069
        model.addAttribute("unreadMap", unreadMap);
1070
        model.addAttribute("lastActivityMap", lastActivityMap);
35640 amit 1071
        model.addAttribute("lastMessageMap", lastMessageMap);
35626 amit 1072
 
34913 ranu 1073
        model.addAttribute("ticketStatusValues", TicketStatus.values());
1074
        model.addAttribute("orderByValues", SortOrder.values());
1075
        model.addAttribute("selectedticketStatus", ticketStatus);
1076
        model.addAttribute("selectedorderby", sortOrder);
1077
        model.addAttribute("ticketSearchTypes", TicketSearchType.values());
1078
        model.addAttribute("ticketSearchType", ticketSearchType);
1079
        model.addAttribute("searchTerm", searchTerm);
35571 amit 1080
        model.addAttribute("authUserListMap", authUserListMap != null ? authUserListMap : new HashMap<>());
34913 ranu 1081
        model.addAttribute("subCategoryIdAndSubCategoryMap", subCategoryIdAndSubCategoryMap);
1082
 
1083
        model.addAttribute("subCategoryIdAndCategoryMap", subCategoryIdAndCategoryMap);
1084
 
1085
        model.addAttribute("activityMap", activityMap);
1086
        model.addAttribute("authUserMap", authUserMap);
1087
        model.addAttribute("activityMapWithActivityId", activityMapWithActivityId);
35569 amit 1088
        boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1089
        model.addAttribute("isCrmUser", isCrmUser);
35626 amit 1090
    }
34913 ranu 1091
 
35626 amit 1092
    @GetMapping(value = "/cs/managerTicket")
1093
    public String getManagerTickets(HttpServletRequest request,
1094
                                    @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
1095
                                    @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
1096
                                    @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
1097
                                    @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
1098
                                    @RequestParam(name = "page", defaultValue = "0") int page,
1099
                                    @RequestParam(name = "pageSize", defaultValue = "25") int pageSize,
1100
                                    @RequestParam(name = "search", required = false) String searchText,
1101
                                    Model model) throws ProfitMandiBusinessException {
1102
        populateManagerTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, page, pageSize, searchText, model);
1103
        return "managerTicket";
34913 ranu 1104
    }
1105
 
35626 amit 1106
    @GetMapping(value = "/cs/managerTicket-content")
1107
    public String getManagerTicketContent(HttpServletRequest request,
1108
                                          @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
1109
                                          @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
1110
                                          @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
1111
                                          @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
1112
                                          @RequestParam(name = "page", defaultValue = "0") int page,
1113
                                          @RequestParam(name = "pageSize", defaultValue = "25") int pageSize,
1114
                                          @RequestParam(name = "search", required = false) String searchText,
1115
                                          Model model) throws ProfitMandiBusinessException {
1116
        populateManagerTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, page, pageSize, searchText, model);
1117
        return "managerTicket-content";
1118
    }
1119
 
1120
    private void populateManagerTicketModel(HttpServletRequest request, SortOrder sortOrder, TicketStatus ticketStatus,
1121
                                            TicketSearchType ticketSearchType, int searchTerm, int page, int pageSize,
1122
                                            String searchText, Model model) throws ProfitMandiBusinessException {
31762 tejbeer 1123
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
35626 amit 1124
        long totalRecords = 0;
31762 tejbeer 1125
        AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
35626 amit 1126
 
1127
        // Block L1-only users from accessing manager tickets
1128
        List<Position> positions = positionRepository.selectAllByAuthUserId(authUser.getId());
1129
        boolean isAboveL1 = positions.stream().anyMatch(pos -> pos.getEscalationType() != EscalationType.L1);
1130
        if (!isAboveL1) {
1131
            throw new ProfitMandiBusinessException("ManagerTicket", 0, "Access denied: requires at least L2 position");
1132
        }
1133
 
35594 amit 1134
        boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1135
 
35626 amit 1136
        List<Integer> userCategoryIds = positionRepository.selectCategoryIdsByAuthUserId(authUser.getId());
1137
 
31762 tejbeer 1138
        List<Ticket> tickets = null;
1139
        Map<Integer, List<AuthUser>> authUserListMap = null;
35594 amit 1140
 
35626 amit 1141
        int offset = page * pageSize;
1142
        Optional<Boolean> resolvedOpt = ticketStatus.equals(TicketStatus.RESOLVED) ? Optional.empty() : Optional.of(TicketStatus.CLOSED.equals(ticketStatus));
1143
 
35594 amit 1144
        if (isCrmUser) {
35626 amit 1145
            tickets = ticketRepository.selectAllTicketsPaginated(
1146
                    resolvedOpt, sortOrder, ticketSearchType, searchTerm, searchText,
1147
                    userCategoryIds, offset, pageSize);
1148
            totalRecords = ticketRepository.selectAllTicketsPaginatedCount(
1149
                    resolvedOpt, ticketSearchType, searchTerm, searchText, userCategoryIds);
31762 tejbeer 1150
        } else {
35626 amit 1151
            tickets = ticketRepository.selectAllManagerTicketPaginated(
1152
                    authUser.getId(), sortOrder, resolvedOpt, ticketSearchType, searchTerm, searchText,
1153
                    offset, pageSize);
1154
            totalRecords = ticketRepository.selectAllCountByManagerTicketPaginated(
1155
                    authUser.getId(), resolvedOpt, ticketSearchType, searchTerm, searchText);
31762 tejbeer 1156
        }
1157
        authUserListMap = csService.getAssignedAuthList(tickets);
1158
 
1159
        if (tickets.size() > 0) {
1160
            Map<Integer, CustomRetailer> fofoIdsAndCustomRetailer = csService.getPartnerByFofoIds(tickets);
1161
            model.addAttribute("fofoIdsAndCustomRetailer", fofoIdsAndCustomRetailer);
1162
        }
1163
 
35626 amit 1164
        int totalPages = (int) Math.ceil((double) totalRecords / pageSize);
1165
        if (totalPages == 0) totalPages = 1;
1166
        int startRecord = totalRecords > 0 ? offset + 1 : 0;
1167
        int endRecord = (int) Math.min(offset + pageSize, totalRecords);
1168
 
1169
        model.addAttribute("size", totalRecords);
1170
        model.addAttribute("totalRecords", totalRecords);
1171
        model.addAttribute("currentPage", page);
1172
        model.addAttribute("pageSize", pageSize);
1173
        model.addAttribute("totalPages", totalPages);
1174
        model.addAttribute("startRecord", startRecord);
1175
        model.addAttribute("endRecord", endRecord);
1176
        model.addAttribute("searchText", searchText != null ? searchText : "");
1177
        model.addAttribute("currentPageDisplay", page + 1);
1178
        model.addAttribute("prevPage", page - 1);
1179
        model.addAttribute("nextPage", page + 1);
1180
        model.addAttribute("lastPage", totalPages - 1);
31762 tejbeer 1181
        model.addAttribute("tickets", tickets);
1182
 
1183
        List<Integer> subCategoryIds = tickets.stream().map(x -> x.getSubCategoryId()).collect(Collectors.toList());
1184
        Map<Integer, TicketSubCategory> subCategoryIdAndSubCategoryMap = csService.getSubCategoryIdAndSubCategoryMap(subCategoryIds);
35571 amit 1185
        if (subCategoryIdAndSubCategoryMap == null) {
1186
            subCategoryIdAndSubCategoryMap = new HashMap<>();
1187
        }
31762 tejbeer 1188
 
1189
        Map<Integer, TicketCategory> subCategoryIdAndCategoryMap = csService.getSubCategoryIdAndCategoryMap(subCategoryIds);
35571 amit 1190
        if (subCategoryIdAndCategoryMap == null) {
1191
            subCategoryIdAndCategoryMap = new HashMap<>();
1192
        }
31762 tejbeer 1193
 
1194
        List<Integer> ticketIds = tickets.stream().map(x -> x.getId()).collect(Collectors.toList());
1195
        Map<Integer, List<Activity>> activityMap = new HashMap<>();
33778 ranu 1196
        Map<Integer, List<Activity>> activityMapWithActivityId = new HashMap<>();
35570 amit 1197
        Map<Integer, AuthUser> authUserMap = new HashMap<>();
31762 tejbeer 1198
 
1199
        if (!ticketIds.isEmpty()) {
35570 amit 1200
            List<Activity> allActivities = activityRepository.selectAll(ticketIds);
1201
            activityMap = allActivities.stream().collect(Collectors.groupingBy(Activity::getTicketId));
1202
            activityMapWithActivityId = allActivities.stream().collect(Collectors.groupingBy(Activity::getId));
31762 tejbeer 1203
 
35570 amit 1204
            Set<Integer> activityCreatorIds = allActivities.stream()
1205
                    .map(Activity::getCreatedBy)
1206
                    .filter(id -> id > 0)
1207
                    .collect(Collectors.toSet());
1208
            if (!activityCreatorIds.isEmpty()) {
1209
                authUserMap = authRepository.selectByIds(new ArrayList<>(activityCreatorIds))
1210
                        .stream().collect(Collectors.toMap(AuthUser::getId, x -> x));
1211
            }
31762 tejbeer 1212
        }
33778 ranu 1213
 
35626 amit 1214
        Map<Integer, Boolean> unreadMap = csService.getUnreadStatusForTickets(tickets, authUser.getId(), UserType.AUTH_USER);
35640 amit 1215
        List<Integer> managerTicketIds = tickets.stream().map(Ticket::getId).collect(Collectors.toList());
1216
        Map<Integer, Activity> lastActivityMap = csService.getLastActivitiesForTickets(managerTicketIds);
1217
        Map<Integer, Activity> lastMessageMap = csService.getLastMessageActivitiesForTickets(managerTicketIds);
35626 amit 1218
        model.addAttribute("unreadMap", unreadMap);
1219
        model.addAttribute("lastActivityMap", lastActivityMap);
35640 amit 1220
        model.addAttribute("lastMessageMap", lastMessageMap);
35626 amit 1221
 
31762 tejbeer 1222
        model.addAttribute("ticketStatusValues", TicketStatus.values());
1223
        model.addAttribute("orderByValues", SortOrder.values());
1224
        model.addAttribute("selectedticketStatus", ticketStatus);
1225
        model.addAttribute("selectedorderby", sortOrder);
1226
        model.addAttribute("ticketSearchTypes", TicketSearchType.values());
1227
        model.addAttribute("ticketSearchType", ticketSearchType);
1228
        model.addAttribute("searchTerm", searchTerm);
1229
        model.addAttribute("authUserListMap", authUserListMap);
1230
        model.addAttribute("subCategoryIdAndSubCategoryMap", subCategoryIdAndSubCategoryMap);
1231
 
1232
        model.addAttribute("subCategoryIdAndCategoryMap", subCategoryIdAndCategoryMap);
1233
 
1234
        model.addAttribute("activityMap", activityMap);
33778 ranu 1235
        model.addAttribute("authUserMap", authUserMap);
1236
        model.addAttribute("activityMapWithActivityId", activityMapWithActivityId);
35594 amit 1237
        model.addAttribute("isCrmUser", isCrmUser);
31762 tejbeer 1238
    }
1239
 
1240
 
1241
    @GetMapping(value = "/cs/edit-ticket")
35594 amit 1242
    public String getEditTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, Model model) throws ProfitMandiBusinessException {
1243
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1244
        AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1245
        boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1246
 
31762 tejbeer 1247
        Ticket ticket = ticketRepository.selectById(ticketId);
35626 amit 1248
        if (ticket == null) {
1249
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
1250
        }
1251
 
1252
        TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectById(ticket.getSubCategoryId());
1253
        int ticketCategoryId = ticketSubCategory.getCategoryId();
1254
        boolean isVisibleSubCategory = ticketSubCategory.isVisibility();
1255
 
1256
        // Check permissions
1257
        boolean hasPositionInCategory = positionRepository.hasCategory(currentAuthUser.getId(), ticketCategoryId);
1258
        List<TicketAssigned> assignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
1259
        boolean isAssigned = assignments.stream().anyMatch(ta -> ta.getAssineeId() == currentAuthUser.getId() || ta.getManagerId() == currentAuthUser.getId());
1260
        boolean isInEscalationChain = ticket.getL1AuthUser() == currentAuthUser.getId() ||
1261
                ticket.getL2AuthUser() == currentAuthUser.getId() ||
1262
                ticket.getL3AuthUser() == currentAuthUser.getId() ||
1263
                ticket.getL4AuthUser() == currentAuthUser.getId() ||
1264
                ticket.getL5AuthUser() == currentAuthUser.getId();
1265
 
1266
        boolean canEdit = (isCrmUser && isVisibleSubCategory)
1267
                || hasPositionInCategory
1268
                || isAssigned
1269
                || isInEscalationChain;
1270
 
1271
        if (!canEdit) {
1272
            throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to edit this ticket");
1273
        }
1274
 
31762 tejbeer 1275
        List<TicketCategory> ticketCategories = csService.getAllTicketCategotyFromSubCategory();
1276
        List<TicketSubCategory> ticketSubCategories = ticketSubCategoryRepository.selectAll(ticketSubCategory.getCategoryId());
1277
        List<AuthUser> authUsers = authRepository.selectAllActiveUser();
35594 amit 1278
 
35617 amit 1279
        // CRM users should not see invisible subcategories
35594 amit 1280
        if (isCrmUser) {
1281
            ticketSubCategories = ticketSubCategories.stream()
35617 amit 1282
                    .filter(TicketSubCategory::isVisibility)
35594 amit 1283
                    .collect(Collectors.toList());
1284
        }
1285
 
1286
        // Get partner name for modal title
1287
        String partnerName = retailerService.getFofoRetailer(ticket.getFofoId()).getBusinessName();
1288
 
31762 tejbeer 1289
        model.addAttribute("ticket", ticket);
1290
        model.addAttribute("ticketCategories", ticketCategories);
1291
        model.addAttribute("ticketSubCategories", ticketSubCategories);
1292
        model.addAttribute("ticketSubCategory", ticketSubCategory);
1293
        model.addAttribute("authUsers", authUsers);
35594 amit 1294
        model.addAttribute("isCrmUser", isCrmUser);
1295
        model.addAttribute("partnerName", partnerName);
31762 tejbeer 1296
        return "edit-ticket-modal";
1297
    }
1298
 
34913 ranu 1299
    @GetMapping(value = "/cs/edit-partner-ticket")
35594 amit 1300
    public String getEditPartnerTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, Model model) throws ProfitMandiBusinessException {
1301
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1302
        AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1303
        boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1304
 
34913 ranu 1305
        Ticket ticket = ticketRepository.selectById(ticketId);
1306
        List<TicketCategory> ticketCategories = csService.getAllTicketCategotyFromSubCategory();
1307
        TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectById(ticket.getSubCategoryId());
1308
        List<TicketSubCategory> ticketSubCategories = ticketSubCategoryRepository.selectAll(ticketSubCategory.getCategoryId());
1309
        List<AuthUser> authUsers = authRepository.selectAllActiveUser();
35594 amit 1310
 
35617 amit 1311
        // CRM users should not see invisible subcategories
35594 amit 1312
        if (isCrmUser) {
1313
            ticketSubCategories = ticketSubCategories.stream()
35617 amit 1314
                    .filter(TicketSubCategory::isVisibility)
35594 amit 1315
                    .collect(Collectors.toList());
1316
        }
1317
 
1318
        // Get partner name for modal title
1319
        String partnerName = retailerService.getFofoRetailer(ticket.getFofoId()).getBusinessName();
1320
 
34913 ranu 1321
        model.addAttribute("ticket", ticket);
1322
        model.addAttribute("ticketCategories", ticketCategories);
1323
        model.addAttribute("ticketSubCategories", ticketSubCategories);
1324
        model.addAttribute("ticketSubCategory", ticketSubCategory);
1325
        model.addAttribute("authUsers", authUsers);
35594 amit 1326
        model.addAttribute("isCrmUser", isCrmUser);
1327
        model.addAttribute("partnerName", partnerName);
34913 ranu 1328
        return "edit-ticket-partner-modal";
1329
    }
1330
 
31762 tejbeer 1331
    @PostMapping(value = "/cs/edit-ticket")
1332
    public String editTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, @RequestParam(name = "subCategoryId", defaultValue = "0") int subCategoryId, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, @RequestParam(name = "authUserId", defaultValue = "0") int authUserId, @RequestParam(name = "escalationType", defaultValue = "L1") EscalationType escalationType, Model model) throws Exception {
1333
        LOGGER.info("Ticket Id {}, CategoryId {}, SubCategory Id {} authUserId {}", ticketId, categoryId, subCategoryId, authUserId);
35592 amit 1334
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1335
 
1336
        // Only admins can edit tickets
1337
        if (!roleManager.isAdmin(loginDetails.getRoleIds())) {
1338
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Only admins can edit tickets");
1339
        }
1340
 
31762 tejbeer 1341
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 1342
        if (ticket == null) {
1343
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
1344
        }
1345
 
35626 amit 1346
        // Verify admin has access to this ticket
35592 amit 1347
        AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1348
        boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1349
 
35626 amit 1350
        // Get ticket's subcategory to check visibility
1351
        TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectById(ticket.getSubCategoryId());
1352
        int ticketCategoryId = ticketSubCategory.getCategoryId();
1353
        boolean isVisibleSubCategory = ticketSubCategory.isVisibility();
35592 amit 1354
 
35626 amit 1355
        // Check if user has position in ticket's category
1356
        boolean hasPositionInCategory = positionRepository.hasCategory(currentAuthUser.getId(), ticketCategoryId);
1357
 
1358
        // Check if assigned or in escalation chain
1359
        List<TicketAssigned> assignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
1360
        boolean isAssigned = assignments.stream().anyMatch(ta -> ta.getAssineeId() == currentAuthUser.getId() || ta.getManagerId() == currentAuthUser.getId());
1361
        boolean isInEscalationChain = ticket.getL1AuthUser() == currentAuthUser.getId() ||
1362
                ticket.getL2AuthUser() == currentAuthUser.getId() ||
1363
                ticket.getL3AuthUser() == currentAuthUser.getId() ||
1364
                ticket.getL4AuthUser() == currentAuthUser.getId() ||
1365
                ticket.getL5AuthUser() == currentAuthUser.getId();
1366
 
1367
        // Permission rules:
1368
        // 1. CRM user AND ticket has visible subcategory, OR
1369
        // 2. Has position in ticket's category (can edit even invisible), OR
1370
        // 3. Is assigned to ticket, OR
1371
        // 4. Is in escalation chain
1372
        boolean canEdit = (isCrmUser && isVisibleSubCategory)
1373
                || hasPositionInCategory
1374
                || isAssigned
1375
                || isInEscalationChain;
1376
 
1377
        if (!canEdit) {
1378
            throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to edit this ticket");
35592 amit 1379
        }
1380
 
31762 tejbeer 1381
        csService.updateTicket(categoryId, subCategoryId, ticket, authUserId, escalationType);
1382
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1383
        return "response";
1384
 
1385
    }
1386
 
34913 ranu 1387
    @PostMapping(value = "/cs/edit-partner-ticket")
1388
    public String editPartnerTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, @RequestParam(name = "subCategoryId", defaultValue = "0") int subCategoryId, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, @RequestParam(name = "authUserId", defaultValue = "0") int authUserId, @RequestParam(name = "escalationType", defaultValue = "L1") EscalationType escalationType, Model model) throws Exception {
1389
        LOGGER.info("Ticket Id {}, CategoryId {}, SubCategory Id {} authUserId {}", ticketId, categoryId, subCategoryId, authUserId);
35592 amit 1390
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1391
 
1392
        // Only admins can edit partner tickets
1393
        if (!roleManager.isAdmin(loginDetails.getRoleIds())) {
1394
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Only admins can edit partner tickets");
1395
        }
1396
 
34913 ranu 1397
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 1398
        if (ticket == null) {
1399
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
1400
        }
1401
 
1402
        // Verify admin has access (must be in Sales, ABM, or RBM position for the partner)
1403
        AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1404
        Map<String, Set<Integer>> authUserPartnerMap = csService.getAuthUserPartnerIdMapping();
1405
        Set<Integer> allowedPartnerIds = authUserPartnerMap.get(currentAuthUser.getEmailId());
1406
 
1407
        if (allowedPartnerIds == null || !allowedPartnerIds.contains(ticket.getFofoId())) {
1408
            // Also allow CRM users
1409
            boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1410
            if (!isCrmUser) {
1411
                throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to edit this partner's ticket");
1412
            }
1413
        }
1414
 
34913 ranu 1415
        csService.updateTicket(categoryId, subCategoryId, ticket, authUserId, escalationType);
1416
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1417
        return "response";
1418
 
1419
    }
1420
 
31762 tejbeer 1421
    @PostMapping(value = "/cs/changeTicketAssignee")
1422
    public String changeTicketAssignee(HttpServletRequest request, @RequestParam(name = "positionId", defaultValue = "0") int positionId, Model model) throws Exception {
1423
        Position position = positionRepository.selectById(positionId);
1424
        if (position.isTicketAssignee()) {
1425
            position.setTicketAssignee(false);
1426
        } else {
1427
            position.setTicketAssignee(true);
1428
        }
1429
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1430
        return "response";
1431
    }
1432
 
1433
 
1434
    @DeleteMapping(value = "/cs/removePosition")
1435
    public String removePosition(HttpServletRequest request, @RequestParam(name = "positionId", defaultValue = "0") int positionId, Model model) throws Exception {
1436
        positionRepository.delete(positionId);
1437
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1438
        return "response";
1439
    }
1440
 
1441
    @PostMapping(value = "/cs/create-last-activity")
1442
    public String createlastActivity(HttpServletRequest request, @RequestParam(name = "ticketId") int ticketId, @RequestParam(name = "lastactivity") ActivityType lastActivity, Model model) throws Exception {
1443
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1444
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 1445
 
1446
        if (ticket == null) {
1447
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
1448
        }
1449
 
1450
        // Authorization check for partners: can only update their own tickets
1451
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
1452
        if (!isAdmin && ticket.getFofoId() != loginDetails.getFofoId()) {
1453
            throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to update this ticket");
1454
        }
1455
 
31762 tejbeer 1456
        Activity activity = new Activity();
1457
        String subject = String.format(ACTIVITY_SUBJECT, ticket.getId());
35592 amit 1458
        if (isAdmin) {
35603 amit 1459
            // CRM team members can resolve any ticket
1460
            // Sales team members can resolve their own Sales category tickets
1461
            // RBM team members can resolve their own RBM category tickets
35592 amit 1462
            AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1463
            boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
35603 amit 1464
            boolean isSalesUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_SALES);
1465
            boolean isRbmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_RBM);
1466
 
1467
            // Check if ticket belongs to Sales or RBM category
1468
            TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectById(ticket.getSubCategoryId());
1469
            boolean isSalesTicket = ticketSubCategory != null && ticketSubCategory.getCategoryId() == ProfitMandiConstants.TICKET_CATEGORY_SALES;
1470
            boolean isRbmTicket = ticketSubCategory != null && ticketSubCategory.getCategoryId() == ProfitMandiConstants.TICKET_CATEGORY_RBM;
1471
 
1472
            // Allow if CRM user OR (Sales user AND Sales ticket) OR (RBM user AND RBM ticket)
1473
            if (!isCrmUser && !(isSalesUser && isSalesTicket) && !(isRbmUser && isRbmTicket)) {
1474
                throw new ProfitMandiBusinessException("Ticket", ticketId, "Only CRM, Sales (for Sales tickets), or RBM (for RBM tickets) team can mark tickets as resolved");
35592 amit 1475
            }
31762 tejbeer 1476
            ticket.setLastActivity(lastActivity);
1477
            String to = retailerService.getFofoRetailer(ticket.getFofoId()).getEmail();
1478
            String message = String.format(PARTNER_RESOLVED_TICKET_MAIL, ticketId, "REOPEN");
1479
            activity.setMessage(message);
35592 amit 1480
            activity.setCreatedBy(authUser.getId());
31762 tejbeer 1481
            activity.setTicketId(ticketId);
1482
            activity.setCreateTimestamp(LocalDateTime.now());
1483
            activity.setType(ActivityType.COMMUNICATION_OUT);
1484
            this.activityRelatedMail(to, null, subject, message);
1485
        } else {
1486
            if (ActivityType.RESOLVED_ACCEPTED == lastActivity) {
1487
                ticket.setLastActivity(lastActivity);
1488
                ticket.setCloseTimestamp(LocalDateTime.now());
1489
                activity.setMessage(ActivityType.RESOLVED_ACCEPTED.toString());
1490
                activity.setCreatedBy(0);
1491
                activity.setTicketId(ticketId);
1492
                activity.setType(ActivityType.COMMUNICATION_IN);
1493
                activity.setCreateTimestamp(LocalDateTime.now());
1494
            } else {
1495
                String message = String.format(INTERNAL_REOPEN_MAIL, ticketId, retailerService.getFofoRetailer(loginDetails.getFofoId()).getBusinessName());
1496
                String to = authRepository.selectById(ticket.getL1AuthUser()).getEmailId();
35395 amit 1497
                String[] ccTo = authRepository.selectByIds(Arrays.asList(ticket.getL2AuthUser(), ticket.getL3AuthUser(), ticket.getL4AuthUser(), ticket.getL5AuthUser())).stream().map(x -> x.getEmailId()).toArray(String[]::new);
31762 tejbeer 1498
                ticket.setLastActivity(lastActivity);
1499
                ticket.setUpdateTimestamp(LocalDateTime.now());
1500
                ticketAssignedRepository.deleteByTicketId(ticketId);
1501
                TicketAssigned ticketAssigned = new TicketAssigned();
1502
                ticketAssigned.setAssineeId(ticket.getL1AuthUser());
1503
                ticketAssigned.setTicketId(ticketId);
1504
                ticketAssignedRepository.persist(ticketAssigned);
1505
                activity.setMessage(INTERNAL_REOPEN_ACTIVITY_MESSAGE);
1506
                activity.setCreatedBy(0);
1507
                activity.setTicketId(ticketId);
1508
                activity.setType(ActivityType.COMMUNICATION_IN);
1509
                activity.setCreateTimestamp(LocalDateTime.now());
1510
                this.activityRelatedMail(to, ccTo, subject, message);
1511
                this.activityRelatedMail(retailerService.getFofoRetailer(loginDetails.getFofoId()).getEmail(), null, subject, String.format(PARTNER_REOPEN, ticketId));
1512
            }
1513
 
1514
        }
1515
        activityRepository.persist(activity);
1516
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1517
        return "response";
1518
    }
1519
 
32812 shampa 1520
 
1521
 
1522
 
1523
 
1524
    @PostMapping(value = "/partner-position/update")
1525
    public String positionUpdated(Model model, @RequestBody List<PartnerPositonUpdateModel> partnerPositionUpdateModels)
1526
            throws Exception {
1527
 
32821 shampa 1528
        Map<Integer, List<String>> positionIdsToAddMap = partnerPositionUpdateModels.stream().filter(x->x.getPositionIdTo()!=0).collect(Collectors.groupingBy(x->x.getPositionIdTo(),
32812 shampa 1529
                Collectors.mapping(x->x.getStoreCode(), Collectors.toList())));
1530
 
32821 shampa 1531
        Map<Integer, List<String>> positionIdsToRemoveMap = partnerPositionUpdateModels.stream().filter(x->x.getPositionIdFrom()!=0).collect(Collectors.groupingBy(x->x.getPositionIdFrom(),
32812 shampa 1532
                Collectors.mapping(x->x.getStoreCode(), Collectors.toList())));
1533
 
1534
        List<Integer> positionIdsToUpdate = new ArrayList<>();
1535
        positionIdsToUpdate.addAll(positionIdsToAddMap.keySet());
1536
        positionIdsToUpdate.addAll(positionIdsToRemoveMap.keySet());
1537
 
35395 amit 1538
        Map<Integer, Position> positionsToUpdateMap =  positionRepository.selectByIds(positionIdsToUpdate).stream().collect(Collectors.toMap(x->x.getId(), x->x));
32812 shampa 1539
        List<Integer> invalidPositionIds = positionsToUpdateMap.values().stream().filter(x-> x.getCategoryId()!= ProfitMandiConstants.TICKET_CATEGORY_RBM
34908 ranu 1540
                && x.getCategoryId() != ProfitMandiConstants.TICKET_CATEGORY_SALES && x.getCategoryId() != ProfitMandiConstants.TICKET_CATEGORY_ABM).map(x -> x.getId()).collect(Collectors.toList());
32812 shampa 1541
        if(invalidPositionIds.size() > 0) {
34908 ranu 1542
            String message = "Non RBM/Sales/ABM are not allowed - " + invalidPositionIds;
32812 shampa 1543
            throw new ProfitMandiBusinessException(message, message, message);
1544
        }
1545
 
1546
        for (Map.Entry<Integer, List<String>> positionIdStoreMapEntry : positionIdsToAddMap.entrySet()) {
1547
            int positionId = positionIdStoreMapEntry.getKey();
1548
            Position position = positionsToUpdateMap.get(positionId);
32821 shampa 1549
            LOGGER.info("positionId - {}, Position - {}", positionId, position);
32812 shampa 1550
            List<String> storeCodesToAdd = positionIdStoreMapEntry.getValue();
1551
            List<Integer> retailerIdsToAdd = fofoStoreRepository.selectByStoreCodes(storeCodesToAdd).stream().map(x->x.getId()).collect(Collectors.toList());
1552
            Map<Integer, PartnerPosition> partnerPositionsMapByFofoId  = partnerPositionRepository
32821 shampa 1553
                    .selectByRegionIdAndPostionId(Arrays.asList(position.getRegionId())
1554
                            ,Arrays.asList(positionId)).stream().collect(Collectors.toMap(x->x.getFofoId(),x->x));
32812 shampa 1555
            for (Integer retailerIdToAdd : retailerIdsToAdd) {
1556
                if (!partnerPositionsMapByFofoId.containsKey(retailerIdToAdd)) {
1557
                    PartnerPosition partnerPositionNew = new PartnerPosition();
1558
                    partnerPositionNew.setPositionId(positionId);
1559
                    partnerPositionNew.setFofoId(retailerIdToAdd);
1560
                    partnerPositionNew.setRegionId(position.getRegionId());
32865 amit.gupta 1561
                    partnerPositionRepository.persist(partnerPositionNew);
32812 shampa 1562
                }
1563
            }
1564
        }
1565
 
1566
        for (Map.Entry<Integer, List<String>> positionIdStoreMapEntry : positionIdsToRemoveMap.entrySet()) {
1567
 
1568
            int positionId = positionIdStoreMapEntry.getKey();
1569
            Position position = positionsToUpdateMap.get(positionId);
1570
            List<String> storeCodesToRemove = positionIdStoreMapEntry.getValue();
1571
            List<Integer> retailerIdsToRemove = fofoStoreRepository.selectByStoreCodes(storeCodesToRemove).stream().map(x->x.getId()).collect(Collectors.toList());
1572
            Map<Integer, PartnerPosition> partnerPositionsMapByFofoId  = partnerPositionRepository
1573
                    .selectByRegionIdAndPostionId(Arrays.asList(position.getRegionId()),Arrays.asList(positionId)).stream().collect(Collectors.toMap(x->x.getFofoId(),x->x));
1574
            for (Integer retailerIdToRemove : retailerIdsToRemove) {
1575
                if (partnerPositionsMapByFofoId.containsKey(retailerIdToRemove)) {
1576
                   PartnerPosition partnerPositionToRemove =  partnerPositionsMapByFofoId.get(retailerIdToRemove);
1577
                   partnerPositionRepository.delete(partnerPositionToRemove);
1578
                }
1579
            }
1580
        }
1581
 
1582
 
1583
 
1584
        /*partnerPositionUpdateModels.str
1585
 
1586
        Map<Integer, Position> positionIdMap = positionsToUpdate.stream().collect(Collectors.toMap(x->x.getId(), x->x));
1587
        for (PartnerPositonUpdateModel partnerPositionUpdateModel : partnerPositionUpdateModels) {
1588
            FofoStore fofoStore = fofoStoreRepository.selectByStoreCode(partnerPositionUpdateModel.getStoreCode());
1589
            Position positionFrom = positionIdMap.get(partnerPositionUpdateModel.getPositionIdFrom());
1590
            Position positionTo = positionIdMap.get(partnerPositionUpdateModel.getPositionIdTo());
1591
            if(positionFrom != null) {
1592
                partnerPositionRepository.selectByRegionIdAndPostionId(Arrays.)
1593
              int regionId = positionFrom.getRegionId()
1594
            }
1595
            if(positionTo != null) {
1596
 
1597
            }
1598
        }*/
1599
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1600
        return "response";
1601
 
1602
    }
1603
 
35626 amit 1604
    @PostMapping(value = "/cs/markTicketRead")
1605
    @ResponseBody
1606
    public Map<String, Object> markTicketRead(HttpServletRequest request,
1607
                                              @RequestParam(name = "ticketId") int ticketId,
1608
                                              Model model) throws ProfitMandiBusinessException {
1609
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1610
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
1611
        Map<String, Object> response = new HashMap<>();
32812 shampa 1612
 
35626 amit 1613
        if (isAdmin) {
1614
            AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1615
            csService.markTicketAsRead(ticketId, authUser.getId(), UserType.AUTH_USER);
1616
        } else {
1617
            csService.markTicketAsRead(ticketId, loginDetails.getFofoId(), UserType.PARTNER);
1618
        }
1619
 
1620
        response.put("success", true);
1621
        return response;
1622
    }
1623
 
1624
    @GetMapping(value = "/cs/unreadCount")
1625
    @ResponseBody
1626
    public Map<String, Object> getUnreadCount(HttpServletRequest request) throws ProfitMandiBusinessException {
1627
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1628
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
1629
        Map<String, Object> response = new HashMap<>();
1630
 
1631
        List<Ticket> tickets;
1632
        int userId;
1633
        UserType userType;
1634
 
1635
        if (isAdmin) {
1636
            AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1637
            userId = authUser.getId();
1638
            userType = UserType.AUTH_USER;
1639
            boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1640
            List<Integer> userCategoryIds = positionRepository.selectCategoryIdsByAuthUserId(authUser.getId());
1641
            if (isCrmUser) {
1642
                tickets = ticketRepository.selectAllTicketsPaginated(
1643
                        Optional.of(false), null, null, 0, null,
1644
                        userCategoryIds, 0, 1000);
1645
            } else {
1646
                tickets = ticketRepository.selectAllByAssigneePaginated(
1647
                        authUser.getId(), Optional.of(false), null, null, 0, null,
1648
                        0, 1000);
1649
            }
1650
        } else {
1651
            userId = loginDetails.getFofoId();
1652
            userType = UserType.PARTNER;
1653
            tickets = ticketRepository.selectAllByCreator(loginDetails.getFofoId(), Optional.of(true), null);
1654
        }
1655
 
1656
        Map<Integer, Boolean> unreadMap = csService.getUnreadStatusForTickets(tickets, userId, userType);
1657
        long unreadCount = unreadMap.values().stream().filter(v -> v).count();
1658
 
1659
        response.put("success", true);
1660
        response.put("unreadCount", unreadCount);
1661
        return response;
1662
    }
1663
 
24417 govind 1664
}