如何覆盖ApplicationTagLib#createLink和g:link的标准行为



背景:我有grails 1.3.7应用程序,它在许多页面上使用g:createLinkg:link

最近,我决定对url映射进行重大更改,引入前面的path元素。

  • 目前我有:/$controller/$action?/$id?
  • 但想要:/$regionId/$controller/$action?/$id?

更改urlMappings很容易,但我不知道如何轻松地更改通过应用程序构建链接的行为。

基本上,我不想浏览每个页面并更改链接。但要在一个地方做到这一点。

问题如何覆盖ApplicationTagLib#createLink功能,以便grails在不需要更改使用此标记(或函数)的页面的情况下使用此实现?

非常感谢任何帮助!

我遇到了一个微笑问题。实际上你可以这样装饰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?"{}

我无法用OOP来解决这个问题。我的意思是,我找不到如何覆盖关闭的方法。我尝试了几种方法,但都没有成功。文档中说,您不能覆盖闭包,只能用新的实现替换它(如果我错了,请纠正我)。

但是(!)我能够通过复制粘贴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
                }
            }
        }
    }
}

将regionId添加到createLink和g:link中的params中,grails足够聪明,可以匹配您的url映射。即

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

最新更新