Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

change LineString facecolor to None #1790

Open
wants to merge 4 commits into
base: main
Choose a base branch
from

Conversation

smartlixx
Copy link
Contributor

Rationale

close #1782 by changing LineString geometries facecolor to None.

Implications

@smartlixx smartlixx force-pushed the FIX_add_LineString_fill branch from f63b366 to ab2ed83 Compare May 17, 2021 12:23
@smartlixx smartlixx force-pushed the FIX_add_LineString_fill branch from ab2ed83 to 7e7a90d Compare May 17, 2021 12:31
@smartlixx smartlixx force-pushed the FIX_add_LineString_fill branch from 7e7a90d to 2cce13d Compare May 17, 2021 12:35
@@ -753,6 +753,20 @@ def add_geometries(self, geoms, crs, **kwargs):

"""
styler = kwargs.pop('styler', None)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What about updating the styler here? You can pass a callable that updates the color that way. You might be able to put the isinstance check in that way and it would act on each geometry then.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK. But maybe it should be clarified that we only do styler update for LineString when
(1) styler is not set (None); or
(2) facecolor is not set, otherwise we should assume the user wants to fill the LineString by explicitly specifying facecolor.

And, what if user passes a callable to styler?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree with all of that. If a user passes a callable for styler then this should have no effect. You could probably add an if callable(styler): return styler in there somewhere.

This should also get some tests to make sure it is covering those cases you mentioned and that we are taking the right paths.

lib/cartopy/mpl/geoaxes.py Outdated Show resolved Hide resolved
@smartlixx smartlixx force-pushed the FIX_add_LineString_fill branch from 951a720 to 6eb951b Compare May 20, 2021 06:47
lib/cartopy/mpl/geoaxes.py Outdated Show resolved Hide resolved
lib/cartopy/mpl/geoaxes.py Outdated Show resolved Hide resolved
Comment on lines 766 to 767
if ('edgecolor' not in kwargs) or (
kwargs['edgecolor'] == 'face'):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
if ('edgecolor' not in kwargs) or (
kwargs['edgecolor'] == 'face'):
if kwargs.get('edgecolor', 'face') == 'face':

But why would 'face' mean patch edgecolor?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Because edgecolor = face means using facecolor value for edge. This suggests the user specifies facecolor, and would like to fill the LineString.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm almost tempted to raise here. Why are we allowing a linestring to be filled at all?

Co-authored-by: Elliott Sales de Andrade <[email protected]>
Copy link
Contributor

@greglucas greglucas left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It has been a while since I looked at this, but it looks like it still needs tests.

Comment on lines 766 to 767
if ('edgecolor' not in kwargs) or (
kwargs['edgecolor'] == 'face'):
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm almost tempted to raise here. Why are we allowing a linestring to be filled at all?

Co-authored-by: Elliott Sales de Andrade <[email protected]>
@smartlixx smartlixx force-pushed the FIX_add_LineString_fill branch from 0f532d6 to 1964e12 Compare November 19, 2021 03:58
@smartlixx
Copy link
Contributor Author

It has been a while since I looked at this, but it looks like it still needs tests.

Before I move on, I have to address the current test error, which I have no clue how to address. Any help would be appreciated.

================================== FAILURES ===================================
_________________ Test_Axes_add_geometries.test_styler_kwarg __________________
self = <cartopy.tests.mpl.test_axes.Test_Axes_add_geometries object at 0x0000005D80019910>
ShapelyFeature = <MagicMock name='ShapelyFeature' id='401595362560'>
add_feature_method = <MagicMock name='add_feature' id='401595271776'>
    @mock.patch('cartopy.mpl.geoaxes.GeoAxes.add_feature')
    @mock.patch('cartopy.feature.ShapelyFeature')
    def test_styler_kwarg(self, ShapelyFeature, add_feature_method):
        ax = GeoAxes(plt.figure(), [0, 0, 1, 1],
                     map_projection=ccrs.Robinson())
        ax.add_geometries(mock.sentinel.geometries, mock.sentinel.crs,
                          styler=mock.sentinel.styler, wibble='wobble')
    
        ShapelyFeature.assert_called_once_with(
            mock.sentinel.geometries, mock.sentinel.crs, wibble='wobble')
    
>       add_feature_method.assert_called_once_with(
            ShapelyFeature(), styler=mock.sentinel.styler)
C:\Miniconda3-x64\envs\test-environment\lib\site-packages\cartopy\tests\mpl\test_axes.py:108: 
_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
C:\Miniconda3-x64\envs\test-environment\lib\unittest\mock.py:919: in assert_called_once_with
    return self.assert_called_with(*args, **kwargs)
_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
self = <MagicMock name='add_feature' id='401595271776'>
args = (<MagicMock name='ShapelyFeature()' id='401595547712'>,)
kwargs = {'styler': sentinel.styler}
expected = call(<MagicMock name='ShapelyFeature()' id='401595547712'>, styler=sentinel.styler)
actual = call(<MagicMock name='ShapelyFeature()' id='401595547712'>, styler=<function GeoAxes.add_geometries.<locals>.styler at 0x0000005D80F8C790>)
_error_message = <function NonCallableMock.assert_called_with.<locals>._error_message at 0x0000005D80F7B790>
cause = None
    def assert_called_with(self, /, *args, **kwargs):
        """assert that the last call was made with the specified arguments.
    
        Raises an AssertionError if the args and keyword args passed in are
        different to the last call to the mock."""
        if self.call_args is None:
            expected = self._format_mock_call_signature(args, kwargs)
            actual = 'not called.'
            error_message = ('expected call not found.\nExpected: %s\nActual: %s'
                    % (expected, actual))
            raise AssertionError(error_message)
    
        def _error_message():
            msg = self._format_mock_failure_message(args, kwargs)
            return msg
        expected = self._call_matcher(_Call((args, kwargs), two=True))
        actual = self._call_matcher(self.call_args)
        if actual != expected:
            cause = expected if isinstance(expected, Exception) else None
>           raise AssertionError(_error_message()) from cause
E           AssertionError: expected call not found.
E           Expected: add_feature(<MagicMock name='ShapelyFeature()' id='401595547712'>, styler=sentinel.styler)
E           Actual: add_feature(<MagicMock name='ShapelyFeature()' id='401595547712'>, styler=<function GeoAxes.add_geometries.<locals>.styler at 0x0000005D80F8C790>)

@greglucas
Copy link
Contributor

I think the problem is that the sentinel isn't callable, so then we go into your if clause and build up the dictionary instead. If you are able to make new tests that test all of the various paths, then you can probably replace this test, or you could potentially change it to a Mock() object that is callable...

>>> import unittest.mock
>>> callable(unittest.mock.sentinel)
False
>>> callable(unittest.mock.Mock())
True

@QuLogic QuLogic modified the milestones: 0.21, 0.22 Sep 11, 2022
@dopplershift dopplershift modified the milestones: 0.22, Next Release Aug 4, 2023
@CLAassistant
Copy link

CLAassistant commented Apr 8, 2024

CLA assistant check
All committers have signed the CLA.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Default styler fills LineStrings
6 participants