- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在使用 angularjs和谷歌柱形图以图表形式显示数据。当数据更多时,我显示带有水平滚动条的图表。滚动条打开时工具提示出现问题。
请查找演示 here .
当鼠标悬停在条形图上时,它会在工具提示中显示条形图的信息。在上面的演示 plunker 中使用水平滚动条滚动到栏的末尾,然后将鼠标悬停以查看问题,工具提示未正确显示。只有在水平滚动后尝试查看数据时才会发生这种情况。
注意到以下属性的问题:
focusTarget: 'category'
如果我删除了上述属性,工具提示会正确显示,但我希望在我的应用程序中使用该属性。有什么建议吗?
样本 javascript代码:
angular.module('myApp', ['googlechart'])
.controller('myController', function($scope) {
var chart1 = {};
chart1.type = "ColumnChart";
chart1.displayed = false;
chart1.data = {
"cols": [{
id: "month",
label: "Month",
type: "string"
}, {
id: "laptop-id",
label: "Laptop",
type: "number"
}, {
id: "desktop-id",
label: "Desktop",
type: "number"
}, {
id: "server-id",
label: "Server",
type: "number"
}, {
id: "cost-id",
label: "Shipping",
type: "number"
}],
"rows": [{
c: [{
v: "January"
}, {
v: 19,
f: "42 items"
}, {
v: 12,
f: "Ony 12 items"
}, {
v: 7,
f: "7 servers"
}, {
v: 4
}]
}, {
c: [{
v: "February"
}, {
v: 13
}, {
v: 1,
f: "1 unit (Out of stock this month)"
}, {
v: 12
}, {
v: 2
}]
}, {
c: [{
v: "March"
}, {
v: 24
}, {
v: 5
}, {
v: 11
}, {
v: 6
}
]
}, {
c: [{
v: "April"
}, {
v: 24
}, {
v: 5
}, {
v: 11
}, {
v: 6
}
]
}, {
c: [{
v: "September"
}, {
v: 4
}, {
v: 2
}, {
v: 51
}, {
v: 6
}
]
}, {
c: [{
v: "October"
}, {
v: 34
}, {
v: 4
}, {
v: 0
}, {
v: 1
}
]
}]
};
chart1.options = {
"title": "Sales per month",
"colors": ['#0000FF', '#009900', '#CC0000', '#DD9900'],
"defaultColors": ['#0000FF', '#009900', '#CC0000', '#DD9900'],
"isStacked": "true",
"fill": 20,
focusTarget: 'category',
"displayExactValues": true,
"vAxis": {
"title": "Sales unit",
"gridlines": {
"count": 10
}
},
"hAxis": {
"title": "Date"
},
"width": chart1.data.rows.length * 130,
"bar": {
groupWidth: 40
}
};
chart1.view = {
columns: [0, 1, 2, 3, 4]
};
$scope.myChart = chart1;
$scope.seriesSelected = function(selectedItem) {
console.log(selectedItem);
var col = selectedItem.column;
if (selectedItem.row === null) {
console.log($scope.myChart.view.columns[col]);
if ($scope.myChart.view.columns[col] == col) {
$scope.myChart.view.columns[col] = {
label: $scope.myChart.data.cols[col].label,
type: $scope.myChart.data.cols[col].type,
calc: function() {
return null;
}
};
$scope.myChart.options.colors[col - 1] = '#CCCCCC';
} else {
console.log("Ran this.");
$scope.myChart.view.columns[col] = col;
console.log($scope.myChart.view.columns[col]);
$scope.myChart.options.colors[col - 1] = $scope.myChart.options.defaultColors[col - 1];
}
}
};
});
最佳答案
你可以包装你的 <div google-chart chart="myChart">
在另一个里面 <div>
具有所需的样式。
<div style="width: 60%; overflow-x: auto; overflow-y: hidden;">
<div google-chart chart="myChart"></div>
</div>
像这样:
(function() {
/**
* @description Google Chart Api Directive Module for AngularJS
* @version 0.0.10
* @author Nicolas Bouillon <nicolas@bouil.org>
* @author GitHub contributors
* @license MIT
* @year 2013
*/
(function(document, window, angular) {
'use strict';
angular.module('googlechart', [])
.value('googleChartApiConfig', {
version: '1',
optionalSettings: {
packages: ['corechart']
}
})
.provider('googleJsapiUrl', function() {
var protocol = 'https:';
var url = '//www.google.com/jsapi';
this.setProtocol = function(newProtocol) {
protocol = newProtocol;
};
this.setUrl = function(newUrl) {
url = newUrl;
};
this.$get = function() {
return (protocol ? protocol : '') + url;
};
})
.factory('googleChartApiPromise', ['$rootScope', '$q', 'googleChartApiConfig', 'googleJsapiUrl', function($rootScope, $q, apiConfig, googleJsapiUrl) {
var apiReady = $q.defer();
var onLoad = function() {
// override callback function
var settings = {
callback: function() {
var oldCb = apiConfig.optionalSettings.callback;
$rootScope.$apply(function() {
apiReady.resolve();
});
if (angular.isFunction(oldCb)) {
oldCb.call(this);
}
}
};
settings = angular.extend({}, apiConfig.optionalSettings, settings);
window.google.load('visualization', apiConfig.version, settings);
};
var head = document.getElementsByTagName('head')[0];
var script = document.createElement('script');
script.setAttribute('type', 'text/javascript');
script.src = googleJsapiUrl;
if (script.addEventListener) { // Standard browsers (including IE9+)
script.addEventListener('load', onLoad, false);
} else { // IE8 and below
script.onreadystatechange = function() {
if (script.readyState === 'loaded' || script.readyState === 'complete') {
script.onreadystatechange = null;
onLoad();
}
};
}
head.appendChild(script);
return apiReady.promise;
}])
.directive('googleChart', ['$timeout', '$window', '$rootScope', 'googleChartApiPromise', function($timeout, $window, $rootScope, googleChartApiPromise) {
return {
restrict: 'A',
scope: {
beforeDraw: '&',
chart: '=chart',
onReady: '&',
onSelect: '&',
select: '&'
},
link: function($scope, $elm, $attrs) {
/* Watches, to refresh the chart when its data, formatters, options, or type change.
All other values intentionally disregarded to avoid double calls to the draw
function. Please avoid making changes to these objects directly from this directive.*/
$scope.$watch(function() {
if ($scope.chart) {
return {
data: $scope.chart.data,
formatters: $scope.chart.formatters,
options: $scope.chart.options,
type: $scope.chart.type,
customFormatters: $scope.chart.customFormatters,
view: $scope.chart.view
};
}
return $scope.chart;
}, function() {
drawAsync();
}, true); // true is for deep object equality checking
// Redraw the chart if the window is resized
var resizeHandler = $rootScope.$on('resizeMsg', function() {
$timeout(function() {
// Not always defined yet in IE so check
if ($scope.chartWrapper) {
drawAsync();
}
});
});
//Cleanup resize handler.
$scope.$on('$destroy', function() {
resizeHandler();
});
// Keeps old formatter configuration to compare against
$scope.oldChartFormatters = {};
function applyFormat(formatType, formatClass, dataTable) {
if (typeof($scope.chart.formatters[formatType]) != 'undefined') {
if (!angular.equals($scope.chart.formatters[formatType], $scope.oldChartFormatters[formatType])) {
$scope.oldChartFormatters[formatType] = $scope.chart.formatters[formatType];
$scope.formatters[formatType] = [];
if (formatType === 'color') {
for (var cIdx = 0; cIdx < $scope.chart.formatters[formatType].length; cIdx++) {
var colorFormat = new formatClass();
for (i = 0; i < $scope.chart.formatters[formatType][cIdx].formats.length; i++) {
var data = $scope.chart.formatters[formatType][cIdx].formats[i];
if (typeof(data.fromBgColor) != 'undefined' && typeof(data.toBgColor) != 'undefined')
colorFormat.addGradientRange(data.from, data.to, data.color, data.fromBgColor, data.toBgColor);
else
colorFormat.addRange(data.from, data.to, data.color, data.bgcolor);
}
$scope.formatters[formatType].push(colorFormat);
}
} else {
for (var i = 0; i < $scope.chart.formatters[formatType].length; i++) {
$scope.formatters[formatType].push(new formatClass(
$scope.chart.formatters[formatType][i]));
}
}
}
//apply formats to dataTable
for (i = 0; i < $scope.formatters[formatType].length; i++) {
if ($scope.chart.formatters[formatType][i].columnNum < dataTable.getNumberOfColumns())
$scope.formatters[formatType][i].format(dataTable, $scope.chart.formatters[formatType][i].columnNum);
}
//Many formatters require HTML tags to display special formatting
if (formatType === 'arrow' || formatType === 'bar' || formatType === 'color')
$scope.chart.options.allowHtml = true;
}
}
function draw() {
if (!draw.triggered && ($scope.chart !== undefined)) {
draw.triggered = true;
$timeout(function() {
if (typeof($scope.chartWrapper) == 'undefined') {
var chartWrapperArgs = {
chartType: $scope.chart.type,
dataTable: $scope.chart.data,
view: $scope.chart.view,
options: $scope.chart.options,
containerId: $elm[0]
};
$scope.chartWrapper = new google.visualization.ChartWrapper(chartWrapperArgs);
google.visualization.events.addListener($scope.chartWrapper, 'ready', function() {
$scope.chart.displayed = true;
$scope.$apply(function(scope) {
scope.onReady({
chartWrapper: scope.chartWrapper
});
});
});
google.visualization.events.addListener($scope.chartWrapper, 'error', function(err) {
console.log("Chart not displayed due to error: " + err.message + ". Full error object follows.");
console.log(err);
});
google.visualization.events.addListener($scope.chartWrapper, 'select', function() {
var selectedItem = $scope.chartWrapper.getChart().getSelection()[0];
$scope.$apply(function() {
if ($attrs.select) {
console.log('Angular-Google-Chart: The \'select\' attribute is deprecated and will be removed in a future release. Please use \'onSelect\'.');
$scope.select({
selectedItem: selectedItem
});
} else {
$scope.onSelect({
selectedItem: selectedItem
});
}
});
});
} else {
$scope.chartWrapper.setChartType($scope.chart.type);
$scope.chartWrapper.setDataTable($scope.chart.data);
$scope.chartWrapper.setView($scope.chart.view);
$scope.chartWrapper.setOptions($scope.chart.options);
}
if (typeof($scope.formatters) === 'undefined')
$scope.formatters = {};
if (typeof($scope.chart.formatters) != 'undefined') {
applyFormat("number", google.visualization.NumberFormat, $scope.chartWrapper.getDataTable());
applyFormat("arrow", google.visualization.ArrowFormat, $scope.chartWrapper.getDataTable());
applyFormat("date", google.visualization.DateFormat, $scope.chartWrapper.getDataTable());
applyFormat("bar", google.visualization.BarFormat, $scope.chartWrapper.getDataTable());
applyFormat("color", google.visualization.ColorFormat, $scope.chartWrapper.getDataTable());
}
var customFormatters = $scope.chart.customFormatters;
if (typeof(customFormatters) != 'undefined') {
for (var name in customFormatters) {
applyFormat(name, customFormatters[name], $scope.chartWrapper.getDataTable());
}
}
$timeout(function() {
$scope.beforeDraw({
chartWrapper: $scope.chartWrapper
});
$scope.chartWrapper.draw();
draw.triggered = false;
});
}, 0, true);
}
}
function drawAsync() {
googleChartApiPromise.then(function() {
draw();
});
}
}
};
}])
.run(['$rootScope', '$window', function($rootScope, $window) {
angular.element($window).bind('resize', function() {
$rootScope.$emit('resizeMsg');
});
}]);
})(document, window, window.angular);
angular.module('myApp', ['googlechart'])
.controller('myController', function($scope) {
var chart1 = {};
chart1.type = "ColumnChart";
chart1.displayed = false;
chart1.data = {
"cols": [{
id: "month",
label: "Month",
type: "string"
}, {
id: "laptop-id",
label: "Laptop",
type: "number"
}, {
id: "desktop-id",
label: "Desktop",
type: "number"
}, {
id: "server-id",
label: "Server",
type: "number"
}, {
id: "cost-id",
label: "Shipping",
type: "number"
}],
"rows": [{
c: [{
v: "January"
}, {
v: 19,
f: "42 items"
}, {
v: 12,
f: "Ony 12 items"
}, {
v: 7,
f: "7 servers"
}, {
v: 4
}]
}, {
c: [{
v: "February"
}, {
v: 13
}, {
v: 1,
f: "1 unit (Out of stock this month)"
}, {
v: 12
}, {
v: 2
}]
}, {
c: [{
v: "March"
}, {
v: 24
}, {
v: 5
}, {
v: 11
}, {
v: 6
}
]
}, {
c: [{
v: "April"
}, {
v: 24
}, {
v: 5
}, {
v: 11
}, {
v: 6
}
]
}, {
c: [{
v: "May"
}, {
v: 18
}, {
v: 11
}, {
v: 7
}, {
v: 2
}
]
}, {
c: [{
v: "June"
}, {
v: 21
}, {
v: 5
}, {
v: 8
}, {
v: 6
}
]
}, {
c: [{
v: "July"
}, {
v: 24
}, {
v: 5
}, {
v: 9
}, {
v: 9
}
]
}, {
c: [{
v: "August"
}, {
v: 14
}, {
v: 1
}, {
v: 11
}, {
v: 5
}
]
}, {
c: [{
v: "September"
}, {
v: 4
}, {
v: 2
}, {
v: 51
}, {
v: 6
}
]
}, {
c: [{
v: "October"
}, {
v: 34
}, {
v: 4
}, {
v: 0
}, {
v: 1
}
]
}]
};
chart1.options = {
"title": "Sales per month",
"colors": ['#0000FF', '#009900', '#CC0000', '#DD9900'],
"defaultColors": ['#0000FF', '#009900', '#CC0000', '#DD9900'],
"isStacked": "true",
"fill": 20,
focusTarget: 'category',
"displayExactValues": true,
"vAxis": {
"title": "Sales unit",
"gridlines": {
"count": 10
}
},
"hAxis": {
"title": "Date"
},
"width": chart1.data.rows.length * 130,
"bar": {
groupWidth: 40
}
};
chart1.view = {
columns: [0, 1, 2, 3, 4]
};
$scope.myChart = chart1;
$scope.seriesSelected = function(selectedItem) {
console.log(selectedItem);
var col = selectedItem.column;
//If there's no row value, user clicked the legend.
if (selectedItem.row === null) {
//If true, the chart series is currently displayed normally. Hide it.
console.log($scope.myChart.view.columns[col]);
if ($scope.myChart.view.columns[col] == col) {
//Replace the integer value with this object initializer.
$scope.myChart.view.columns[col] = {
//Take the label value and type from the existing column.
label: $scope.myChart.data.cols[col].label,
type: $scope.myChart.data.cols[col].type,
//makes the new column a calculated column based on a function that returns null,
//effectively hiding the series.
calc: function() {
return null;
}
};
//Change the series color to grey to indicate that it is hidden.
//Uses color[col-1] instead of colors[col] because the domain column (in my case the date values)
//does not need a color value.
$scope.myChart.options.colors[col - 1] = '#CCCCCC';
}
//series is currently hidden, bring it back.
else {
console.log("Ran this.");
//Simply reassigning the integer column index value removes the calculated column.
$scope.myChart.view.columns[col] = col;
console.log($scope.myChart.view.columns[col]);
//I had the original colors already backed up in another array. If you want to do this in a more
//dynamic way (say if the user could change colors for example), then you'd need to have them backed
//up when you switch to grey.
$scope.myChart.options.colors[col - 1] = $scope.myChart.options.defaultColors[col - 1];
}
}
};
});
})();
<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.3.0-beta.18/angular.js"></script>
<div data-ng-app="myApp">
<div data-ng-controller="myController">
<div style="width: 60%; overflow-x: auto; overflow-y: hidden;">
<div data-google-chart data-chart="myChart"></div>
</div>
<br><br>
<div data-google-chart data-chart="myChart"></div>
</div>
</div>
关于javascript - onmouseover 工具提示未正确显示,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45804673/
我的 friend 编写了一个程序,它比较随机排列的骰子面,以找到分布最均匀的面——尤其是当面不仅仅是序列时。 我将他的程序翻译成 haskell 是因为我一直在寻找一个理由来让别人知道 haskel
我需要对表单中的某些字段进行评论/提示。我的想法是在模型中描述它,就像attributeLabels一样。我该怎么做? 然后它会是理想的,如果 Gii 模型(和 Crud)生成器直接从 mysql 列
我们使用 FastReport 来生成报告。事实上,我们为访问源代码付费。 我们目前使用的是 FastReport 的最新稳定版本。虽然它对于我们的生产来说足够稳定,但每当我编译时,我都会看到以下内容
我需要创建一个对话框/提示,包括用于用户输入的文本框。我的问题是,确认对话框后如何获取文本?通常我会为此创建一个类,将文本保存在属性中。不过我想使用 XAML 设计对话框。因此,我必须以某种方式扩展
我想提示用户是否要执行操作(删除) - 用警报框说"is"或“否”,如果是,则运行删除脚本,如果否,则不执行任何操作 我不太了解 javascript,因此是否有人可以使用 javascript 获得
所以我正在编写一个简单的 JS 代码。我们刚刚开始学习函数。我需要创建一个名为“printStars”的函数。 我需要从用户那里获取一个号码,并根据该号码打印“*”。 这就是我所做的:
我在我的页面上添加了一个提示,但它在页面加载之前加载了。如何仅在整个页面可见时才显示消息? 这是我的提示: if (name == null || name == "") { txt == "No
我在我的页面上添加了一个提示,但它在页面加载之前加载了。如何仅在整个页面可见时才显示消息? 这是我的提示: if (name == null || name == "") { txt == "No
我正在自定义我的 zsh 提示,并发现以下内容来检查是否有任何后台作业: if [[ $(jobs | wc -l) -gt 0 ]]; then # has background job(s)
这个问题在这里已经有了答案: JavaScript object: access variable property by name as string [duplicate] (3 个答案) pa
我正在尝试用 javascript 制作一个简单的数学练习程序。在提示警报中给出不同的值,并将答案与用户输入进行比较。这是代码: Calculations generate(); functio
在这段代码中,尽管我使用了文本对齐属性在“编辑文本” View 的中心设置“提示”。但它无法正常工作。 最佳答案 尝试 关于android - 如何在编辑文本的中心对齐文本(提示),我们在Sta
我正在尝试让我的 EditText 显示一个提示,例如“请在此处输入答案”,当用户点击 EditText 以键入他们的答案时,文本应该消失并留空,以便他们在其中输入答案. 截至目前,这就是我的 .xm
我当前的 android 应用程序中有两个微调器,我想要一个默认值,例如 editText 的 android:hint 功能。有没有办法这样做,但不会将提示添加到填充微调器的字符串数组。例如从微调器
如果我的表单已完全填写,我如何提示“感谢您填写表单,“name”!” function submit_onclick() { if(confirm("Thanks for completing t
我刚刚了解了prompt()命令;我知道 Prompt() 命令以字符串的形式返回用户输入。我正在搞乱下面的程序,我输入了Per“Dead”Ohlin作为男性名字。为什么这有效并且没有引起任何问题?
void openUpNow(FILE *x, FILE *y) { x = fopen("xwhatever", "r"); y = fopen("ywhatever", "r");
我有一个作业正在处理,但我在使用 prompt() 方法时遇到了问题。我看到我可以做一个提示,但我需要几个并且有数量。 例如... 我创建了一个 HTML 表格,其中包含许多艺术家和包含 DVD、CD
我正在学习 Big Nerd Ranch 的 iOS Programming, 2nd Edition,我已经来到第 4 章挑战:标题。该练习暗示我感到困惑;它说我需要做一些我认为不需要做的事情。 到
抱歉,如果这是微不足道的,但我没有找到任何解决此问题的建议。我在 Ubuntu 上,我的 Yii 项目需要 PHPUnit。我已经安装了 PHPUnit 两次,方法是下载 phpunit.phar 并
我是一名优秀的程序员,十分优秀!