Personalización de directivas ngTagsInput y autoComplete (AngularJS)
Soy nuevo en AngularJS y actualmente estoy trabajando en un campo de entrada, que puede aceptar varias etiquetas a la vez junto con la función de autocompletar, que muestra las etiquetas disponibles como opciones desplegables. Para esto estoy usando elngTagsInput
directiva que encontré en la web (http://mbenford.github.io/ngTagsInput/), que me da un elemento HTML personalizado<tags-input>
. Esto funciona de maravilla:
index.html
:
<script>
var app = angular.module('plunker', ['ngTagsInput']);
app.controller('MainCtrl', function($scope, $http) {
$scope.tags = [
{ text: 'Tag1' },
{ text: 'Tag2' },
{ text: 'Tag3' }
];
$scope.loadTags = function(query) {
return $http.get('tags.json');
};
});
</script>
<div ng-app="plunker" ng-controller="MainCtrl">
<tags-input ng-model="tags" add-on-paste="true" display-property="text" placeholder="Add a Tag" add-from-autocomplete-only="true">
<auto-complete max-results-to-show="4" min-length="2" source="loadTags($query)"></auto-complete>
</tags-input>
</div>
tags.json
:
[
{ "text": "Tag1" },
{ "text": "Tag2" },
{ "text": "Tag3" },
{ "text": "Tag4" },
{ "text": "Tag5" },
{ "text": "Tag6" },
{ "text": "Tag7" },
{ "text": "Tag8" },
{ "text": "Tag9" },
{ "text": "Tag10" }
]
Sin embargo, quería usar el HTML estándar<input>
elemento en lugar de la costumbre<tags-input>
elemento que viene junto con la directiva, por lo que con mucha ayuda y uso<script src="https://code.jquery.com/jquery-3.1.0.js" integrity="sha256-slogkvB1K3VOkzAI8QITxV3VzpOnkeNVsKvtkYLMjfk=" crossorigin="anonymous"></script>
Pude hacerlo aquí:
Aquí está lo nuevoindex.html
:
<script>
var app = angular.module('plunker', ['ngTagsInput']);
app.controller('MainCtrl', function($scope, $http) {
$scope.tags = [
{ "id":1, "tagname": 'Tag1' },
{ "id":2, "tagname": 'Tag2' },
{ "id":3, "tagname": 'Tag3' },
{ "id":4, "tagname": 'Tag4' }
];
$scope.loadTags = function(query) {
return $http.get('tags.json');
};
});
app.directive('tagsInputAttr',
function($compile){
return {
restrict: 'A',
require: '?ngModel',
scope:{
ngModel: '='
},
link: function($scope, element, attrs, controller) {
var attrsText = '';
$.each($(element)[0].attributes, function(idx, attr) {
if (attr.nodeName === "tags-input-attr" || attr.nodeName === "ng-model")
return;
attrsText += " " + attr.nodeName + "='" + attr.nodeValue + "'";
});
var html ='<tags-input ng-model="ngModel" ' + attrsText + '></tags-input>';
e =$compile(html)($scope);
$(element).replaceWith(e);
}
};
}
);
</script>
<div ng-app="plunker" ng-controller="MainCtrl">
<input tags-input-attr ng-model="tags" add-on-paste="true" display-property="tagname" placeholder="Add tags here..." add-from-autocomplete-only="true">
<auto-complete max-results-to-show="3" min-length="2" source="loadTags($query)"></auto-complete>
</input>
</div>
Y lo nuevotags.json
:
[
{ "id":1, "tagname": "Tag1" },
{ "id":2, "tagname": "Tag2" },
{ "id":3, "tagname": "Tag3" },
{ "id":4, "tagname": "Tag4" },
{ "id":5, "tagname": "Tag5" },
{ "id":6, "tagname": "Tag6" },
{ "id":7, "tagname": "Tag7" },
{ "id":8, "tagname": "Tag8" },
{ "id":9, "tagname": "Tag9" },
{ "id":10, "tagname": "Tag10" }
]
Como puede observar, la nueva directivatagsInputAttr
, que envuelve el<tags-input>
proporciona la misma funcionalidad y se puede usar dentro<input>
etiqueta como un atributo junto con el resto de atributos comong-model
, display-property
etc. Así que no tengo que usar el<tags-input>
elemento directamente. losproblema es que el<auto-complete>
colocado dentro del<input>
La etiqueta no funciona.
Para esto necesito alterar mi directiva, considerando lo siguiente:
Nota: No quiero usar jquery para esto
Mi pregunta es cómo envuelvo el<auto-complete>
dentro del mismo<input tags-input-attr>
elemento:
Ya sea como un atributo dentro del mismo<input tags-input-attr>
elemento
o como un atributo dentro de un elemento HTML estándar como<div>
o<span>
, envuelto dentro del mismo<input tags-input-attr>
elemento.
Si no son los dos anteriores, entonces como último recurso, como el<auto-complete>
etiqueta envuelta dentro de la misma<input tags-input-attr>
elemento
Toda ayuda es apreciada. Gracias por adelantado.