Как переопределить стандартное поведение ApplicationTagLib # createLink и g: link?

Фон: У меня есть приложение grails 1.3.7, которое использует g:createLink и g:link на многих страницах.

Недавно я решил сделать большое изменение в сопоставлениях url - ввести предыдущий элемент пути.

  • В настоящее время у меня есть: /$controller/$action?/$id?
  • Но хочу иметь: /$regionId/$controller/$action?/$id?

Было легко изменить urlMappings, но я не могу понять, как легко изменить поведение способов создания ссылок в приложении.

В принципе, я не хочу проходить через каждую страницу и менять ссылки. Но хотите сделать это в одном месте.

Вопрос Как переопределить функциональность ApplicationTagLib # createLink, чтобы grails использовала эту реализацию без необходимости внесения изменений в страницы, использующие этот тег (или функцию)?

Любая помощь сильно оценивается!

Ответ 1

Мне не удалось решить эту проблему с точки зрения ООП. Я имею в виду, что я не могу найти способ переопределить закрытие. Я пробовал несколько подходов, но безуспешно. И документация говорит, что вы не можете переопределить закрытие, вы можете заменить ее только новой реализацией (пожалуйста, исправьте меня, если я ошибаюсь).

Но (!) Я смог решить задачу, скопировав исходный код метода ApplicationTagLib # createLink. Я думаю, что это жестокое решение, но после 8 часов борьбы с этой простой задачей - это приемлемо.

Итак, наконец, все, что мне нужно сделать - это определить этот класс, grails сразу же будет использовать его для генерации ссылок (для всех просмотров, не нужно менять их код):

import java.text.SimpleDateFormat;

import groovy.time.*;
import java.text.*;

import org.codehaus.groovy.grails.commons.GrailsControllerClass
import org.codehaus.groovy.grails.plugins.web.taglib.ApplicationTagLib;
import org.codehaus.groovy.grails.web.mapping.UrlCreator
import org.codehaus.groovy.grails.commons.ControllerArtefactHandler
import org.springframework.web.context.request.RequestContextHolder

class OverrideTagLib extends ApplicationTagLib {

    def createLink = { attrs ->
        // get value for regionId parameter
        def regionId = regionIdFinderService.currentRegionId

        // add cutsom regionId parameter
        if (attrs) {
            if (attrs.params)
                attrs.params.put("regionId", regionId);
            else {
                attrs.params = ["regionId":regionId];
            }
        }

        // process
        def writer = getOut()
        // prefer URI attribute
        if (attrs.uri) {
            writer << handleAbsolute(attrs)
            writer << attrs.uri.toString()
        }
        else {
            // prefer a URL attribute
            def urlAttrs = attrs
            if (attrs.url instanceof Map) {
                urlAttrs = attrs.remove('url').clone()
            }
            else if (attrs.url) {
                urlAttrs = attrs.remove('url').toString()
            }

            if (urlAttrs instanceof String) {
                if (useJsessionId) {
                    writer << response.encodeURL(urlAttrs)
                }
                else {
                    writer << urlAttrs
                }
            }
            else {
                def controller = urlAttrs.containsKey("controller") ? urlAttrs.remove("controller")?.toString() : controllerName
                def action = urlAttrs.remove("action")?.toString()
                if (controller && !action) {
                    GrailsControllerClass controllerClass = grailsApplication.getArtefactByLogicalPropertyName(ControllerArtefactHandler.TYPE, controller)
                    String defaultAction = controllerClass?.getDefaultAction()
                    if (controllerClass?.hasProperty(defaultAction)) {
                        action = defaultAction
                    }
                }
                def id = urlAttrs.remove("id")
                def frag = urlAttrs.remove('fragment')?.toString()
                def params = urlAttrs.params && urlAttrs.params instanceof Map ? urlAttrs.remove('params') : [:]
                def mappingName = urlAttrs.remove('mapping')
                if (mappingName != null) {
                    params.mappingName = mappingName
                }
                if (request['flowExecutionKey']) {
                    params."execution" = request['flowExecutionKey']
                }

                if (urlAttrs.event) {
                    params."_eventId" = urlAttrs.remove('event')
                }
                def url
                if (id != null) params.id = id
                def urlMappings = applicationContext.getBean("grailsUrlMappingsHolder")
                UrlCreator mapping = urlMappings.getReverseMapping(controller,action,params)

                // cannot use jsessionid with absolute links
                if (useJsessionId && !attrs.absolute) {
                    url = mapping.createURL(controller, action, params, request.characterEncoding, frag)
                    def base = attrs.remove('base')
                    if (base) writer << base
                    writer << response.encodeURL(url)
                }
                else {
                    url = mapping.createRelativeURL(controller, action, params, request.characterEncoding, frag)
                    writer << handleAbsolute(attrs)
                    writer << url
                }
            }
        }
    }
}

Ответ 2

У меня была знакомая проблема. На самом деле вы можете украсить тег g: link следующим образом.

1) TagLib

import org.codehaus.groovy.grails.plugins.web.taglib.*

class OverrideDefaultTagLib {

    static namespace = "g"

    def link = { attrs, body ->

        def c = "1" // Get it from session or somewhere else

        if (attrs.params) {
            attrs.params.put("region", c)
        } else {
            attrs.params = [region: c]
        }

        def applicationTagLib = grailsApplication.mainContext.getBean('org.codehaus.groovy.grails.plugins.web.taglib.ApplicationTagLib')
            applicationTagLib.link.call(attrs, body)
        }
    }
}

2) добавить в UrlMappings.groovy

"/$region/$controller/$action?/$id?"{}

Ответ 3

добавить regionId к параметрам в createLink и g: link и grails достаточно умен, чтобы соответствовать вашим urlmappings. то есть

${createLink(controller:'c',action:'a',id:1,params:[regionId:2])}