如何防止重定向<a href="#something"> in Angular Js1.2</a>

时间:2013-11-15 06:46:13

标签: angularjs twitter-bootstrap migration

我在我的应用程序中使用了AngularJs-1.0.7和Bootstrap。最近我从AngularJs-1.0.7迁移到AngularJs-1.2。我正在使用Bootstrap的Accordions和Tabs。

Tab的Html代码包含<a href="#id_for_content">,如下所示。

<ul id="myTab" class="nav nav-tabs">
    <li class="active"><a href="#firstTab" data-toggle="tab">Home</a></li>
    <li><a href="#secondTab" data-toggle="tab">Profile</a></li>
</ul>

<div id="myTabContent" class="tab-content">
    <div class="tab-pane fade in active" id="firstTab">
      <p>Content for first tab.</p>
    </div>
    <div class="tab-pane fade" id="secondTab">
      <p>Content For second tab.</p>
    </div>
</div>

在Angular的旧版本中,只有在我们提供<a href="#/firstTab">之类的锚标记时才会发生路线更改。但AngularJs-1.2重定向<a href="#firstTab">。它不考虑/#之间的firstTab。因此,在单击Tab时,它会重定向到http://web_url/#/firstTab。如何解决这个问题?


我的解决方案

我找到了解决此问题的方法。我为标签写了一个指令。在该指令中,我检查了href属性。如果匹配阻止其默认行为。请检查以下代码。

app.directive('a', function() {
    return {
        restrict: 'E',
        link: function(scope, elem, attrs) {
            if(attrs.href === '#firstTab'|| attrs.href === '#secondTab'){
                elem.on('click', function(e){
                    e.preventDefault();
                });
            }
        }
   };
}); 

但是这个方法的问题是,我必须在这里检查每个标签id或手风琴ID。如果我为它们使用动态ID,则无法检入指令。

如果您能找到更好的解决方案,请让我们都知道。

9 个答案:

答案 0 :(得分:47)

只需将其添加到您的链接:

target="_self"

你准备好了;)

答案 1 :(得分:19)

尝试data-target="#something",它在开发和生产环境中都适合我。

答案 2 :(得分:11)

我想告诉您,还有另一种解决方案可以解决此问题(无需在控制器中定义功能)。

您可以创建一个指令来阻止默认浏览器的行为(实际上是在更改URL):

var PreventDefault = function () {

    var linkFn = function (scope, element, attrs) {
        $(element).on("click", function (event){
            event.preventDefault();
        });
    };

    return {
        restrict: 'A',
        link: linkFn
    }
};

然后只需将指令添加到负责切换选项卡的每个a元素:

<ul id="myTab" class="nav nav-tabs">
    <li class="active"><a href="#firstTab" prevent-default data-toggle="tab">Home</a></li>
    <li><a href="#secondTab" prevent-default data-toggle="tab">Profile</a></li>
</ul>

<div id="myTabContent" class="tab-content">
    <div class="tab-pane fade in active" id="firstTab">
      <p>Content for first tab.</p>
    </div>
    <div class="tab-pane fade" id="secondTab">
      <p>Content For second tab.</p>
    </div>
</div>

答案 3 :(得分:6)

通常,您可以使用rootscope在所有范围的原型上定义函数。添加ng-click你的锚标签并将$ event 传递给rootscope处理程序。 Angular也可以通过ng-click发送原始事件:

<a ng-click="tabclick($event)">

在模块的运行块中,您可以在$ rootScope上设置tabClick函数

$rootScope.tabclick = function ($event) { $event.preventDefault(); }

这适用于任何范围,您无需检查特定的元素ID。

示例小提琴:http://jsfiddle.net/stto3703/wQe6P/

干杯

答案 4 :(得分:2)

您可以使用ng-click调用JS中的函数。在该功能中,使用事件的preventDefault方法,并使用$location的{​​{1}}方法更改网址中的哈希值。

答案 5 :(得分:1)

我一直有这个问题已经有一段时间了,我只是偶然发现了这个问题(以及你自己的答案)。事实证明,你的解决方案几乎是正确的,只需稍加修改就可以按照你的意愿工作。

除了你使用标签的问题之外,我还遇到了下拉链接的这个问题 - 他们在点击时导航离开当前页面,而不是下拉菜单 - 事实上,这个问题存在于任何旨在切换内容的链接上,即设置了数据切换属性。

因此,对代码进行略微修改以检查是否存在“toggle”属性可以解决这个问题:

app.directive('a', function() {
    return {
        restrict: 'E',
        link: function(scope, elem, attrs) {
            if(attrs.toggle){
                elem.on('click', function(e){
                    e.preventDefault();
                });
            }
        }
   };
});

希望这有帮助!

答案 6 :(得分:0)

您问题中的解决方案确实有效,但为了防止必须检查每个主播的ID,请更改

if (attrs.href === '#firstTab'|| attrs.href === '#secondTab')

if (attrs.href && attrs.href.indexOf('#') > -1)

指令:

.directive('a', function () {
    return {
        restrict: 'E',
        link: function (scope, elem, attrs) {
            if (attrs.href && attrs.href.indexOf('#') > -1) {
                elem.on('click', function (e) {
                    e.preventDefault();
                });
            }
        }
    };
})  

答案 7 :(得分:0)

信用额度 https://prerender.io/js-seo/angularjs-seo-get-your-site-indexed-and-to-the-top-of-the-search-results/

我的解决方案:   - 在app.js内部尝试添加“$ locationProvider.hashPrefix('!');”

    .config(['$stateProvider', '$urlRouterProvider', '$httpProvider', '$locationProvider',
    function($stateProvider, $urlRouterProvider, $httpProvider, $locationProvider) {
    $stateProvider
    .state('menu', {
        url: '/',
        cache: false,
        abstract: true,
        templateUrl: 'static/www/templates/menu.html'
    })
    .state('menu.main_page', {
    url: 'app/main',
    cache: false,
    views: {
        'menuContent': {
          templateUrl: 'static/www/templates/mainpage.html',
          controller: 'MainCtrl'
        }
    }
    })
    $locationProvider.hashPrefix('!');
    $urlRouterProvider.otherwise(function ($injector, $location) {
        var $state = $injector.get('$state');
        $state.go('menu.mainpage');
    });
   }])
    在index.html里面
  • 尝试添加

    <meta name="fragment" content="!">

Marcel 上面说的

.directive('a', function () {
return {
    restrict: 'E',
    link: function (scope, elem, attrs) {
        if (attrs.href && attrs.href.indexOf('#') > -1) {
            elem.on('click', function (e) {
                e.preventDefault();
            });
        }
    }
};
})

就是这样!适合我!

答案 8 :(得分:0)

如果您之前的答案没有成功,您可以尝试使用$ timeout ...在这种情况下,我使用了AngularUI标签......

vm.removeUserTab = function (user) {
    $timeout(function () {
        vm.activeTab = 0;
        var index = vm.userTabs.indexOf(user);
        vm.userTabs.splice(index, 1);
    });
};