├── .gitignore ├── LICENSE ├── Procfile ├── README.md ├── app.py ├── data_explorer.py ├── geemap_app.py ├── requirements.txt ├── setup.sh └── streamlit_app.py /.gitignore: -------------------------------------------------------------------------------- 1 | # Byte-compiled / optimized / DLL files 2 | __pycache__/ 3 | *.py[cod] 4 | *$py.class 5 | 6 | # C extensions 7 | *.so 8 | .vscode/ 9 | 10 | # Distribution / packaging 11 | .Python 12 | build/ 13 | develop-eggs/ 14 | dist/ 15 | downloads/ 16 | eggs/ 17 | .eggs/ 18 | lib/ 19 | lib64/ 20 | parts/ 21 | sdist/ 22 | var/ 23 | wheels/ 24 | pip-wheel-metadata/ 25 | share/python-wheels/ 26 | *.egg-info/ 27 | .installed.cfg 28 | *.egg 29 | MANIFEST 30 | 31 | # PyInstaller 32 | # Usually these files are written by a python script from a template 33 | # before PyInstaller builds the exe, so as to inject date/other infos into it. 34 | *.manifest 35 | *.spec 36 | 37 | # Installer logs 38 | pip-log.txt 39 | pip-delete-this-directory.txt 40 | 41 | # Unit test / coverage reports 42 | htmlcov/ 43 | .tox/ 44 | .nox/ 45 | .coverage 46 | .coverage.* 47 | .cache 48 | nosetests.xml 49 | coverage.xml 50 | *.cover 51 | *.py,cover 52 | .hypothesis/ 53 | .pytest_cache/ 54 | 55 | # Translations 56 | *.mo 57 | *.pot 58 | 59 | # Django stuff: 60 | *.log 61 | local_settings.py 62 | db.sqlite3 63 | db.sqlite3-journal 64 | 65 | # Flask stuff: 66 | instance/ 67 | .webassets-cache 68 | 69 | # Scrapy stuff: 70 | .scrapy 71 | 72 | # Sphinx documentation 73 | docs/_build/ 74 | 75 | # PyBuilder 76 | target/ 77 | 78 | # Jupyter Notebook 79 | .ipynb_checkpoints 80 | 81 | # IPython 82 | profile_default/ 83 | ipython_config.py 84 | 85 | # pyenv 86 | .python-version 87 | 88 | # pipenv 89 | # According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. 90 | # However, in case of collaboration, if having platform-specific dependencies or dependencies 91 | # having no cross-platform support, pipenv may install dependencies that don't work, or not 92 | # install all needed dependencies. 93 | #Pipfile.lock 94 | 95 | # PEP 582; used by e.g. github.com/David-OConnor/pyflow 96 | __pypackages__/ 97 | 98 | # Celery stuff 99 | celerybeat-schedule 100 | celerybeat.pid 101 | 102 | # SageMath parsed files 103 | *.sage.py 104 | 105 | # Environments 106 | .env 107 | .venv 108 | env/ 109 | venv/ 110 | ENV/ 111 | env.bak/ 112 | venv.bak/ 113 | 114 | # Spyder project settings 115 | .spyderproject 116 | .spyproject 117 | 118 | # Rope project settings 119 | .ropeproject 120 | 121 | # mkdocs documentation 122 | /site 123 | 124 | # mypy 125 | .mypy_cache/ 126 | .dmypy.json 127 | dmypy.json 128 | 129 | # Pyre type checker 130 | .pyre/ 131 | .streamlit/secrets.toml 132 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2020 Qiusheng Wu 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /Procfile: -------------------------------------------------------------------------------- 1 | web: sh setup.sh && streamlit run app.py -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # geemap-streamlit 2 | 3 | 4 | To deploy an Earth Engine App with streamlit, one needs to add an environment variable `EARTHENGINE_TOKEN` to streamlit. See [Secrets Management](https://www.notion.so/Secrets-Management-730c82af2fc048d383d668c4049fb9bf) for details. 5 | 6 | ![](https://i.imgur.com/9ifDaJo.png) -------------------------------------------------------------------------------- /app.py: -------------------------------------------------------------------------------- 1 | import streamlit as st 2 | from streamlit_folium import folium_static 3 | import folium 4 | 5 | "# streamlit-geemap" 6 | 7 | with st.echo(): 8 | import streamlit as st 9 | from streamlit_folium import folium_static 10 | import ee 11 | import geemap.eefolium as geemap 12 | 13 | m = geemap.Map() 14 | 15 | dem = ee.Image('USGS/SRTMGL1_003') 16 | # Set visualization parameters. 17 | vis_params = { 18 | 'min': 0, 19 | 'max': 4000, 20 | 'palette': ['006633', 'E5FFCC', '662A00', 'D8D8D8', 'F5F5F5']} 21 | 22 | m.addLayer(dem, vis_params, 'DEM') 23 | m.addLayerControl() 24 | 25 | # call to render Folium map in Streamlit 26 | folium_static(m) 27 | -------------------------------------------------------------------------------- /data_explorer.py: -------------------------------------------------------------------------------- 1 | import streamlit as st 2 | import pandas as pd 3 | import numpy as np 4 | 5 | st.title('Uber pickups in NYC') 6 | 7 | DATE_COLUMN = 'date/time' 8 | DATA_URL = ('https://s3-us-west-2.amazonaws.com/' 9 | 'streamlit-demo-data/uber-raw-data-sep14.csv.gz') 10 | 11 | @st.cache 12 | def load_data(nrows): 13 | data = pd.read_csv(DATA_URL, nrows=nrows) 14 | lowercase = lambda x: str(x).lower() 15 | data.rename(lowercase, axis='columns', inplace=True) 16 | data[DATE_COLUMN] = pd.to_datetime(data[DATE_COLUMN]) 17 | return data 18 | 19 | data_load_state = st.text('Loading data...') 20 | data = load_data(10000) 21 | data_load_state.text("Done! (using st.cache)") 22 | 23 | if st.checkbox('Show raw data'): 24 | st.subheader('Raw data') 25 | st.write(data) 26 | 27 | st.subheader('Number of pickups by hour') 28 | hist_values = np.histogram(data[DATE_COLUMN].dt.hour, bins=24, range=(0,24))[0] 29 | st.bar_chart(hist_values) 30 | 31 | # Some number in the range 0-23 32 | hour_to_filter = st.slider('hour', 0, 23, 17) 33 | filtered_data = data[data[DATE_COLUMN].dt.hour == hour_to_filter] 34 | 35 | st.subheader('Map of all pickups at %s:00' % hour_to_filter) 36 | st.map(filtered_data) -------------------------------------------------------------------------------- /geemap_app.py: -------------------------------------------------------------------------------- 1 | import streamlit as st 2 | from streamlit_folium import folium_static 3 | import geemap.eefolium as geemap 4 | import ee 5 | 6 | # os.environ["EARTHENGINE_TOKEN"] == st.secrets["EARTHENGINE_TOKEN"] 7 | 8 | "# streamlit geemap demo" 9 | st.markdown('Source code: ') 10 | 11 | with st.echo(): 12 | import streamlit as st 13 | from streamlit_folium import folium_static 14 | import geemap.eefolium as geemap 15 | import ee 16 | 17 | m = geemap.Map() 18 | dem = ee.Image('USGS/SRTMGL1_003') 19 | 20 | vis_params = { 21 | 'min': 0, 22 | 'max': 4000, 23 | 'palette': ['006633', 'E5FFCC', '662A00', 'D8D8D8', 'F5F5F5']} 24 | 25 | m.addLayer(dem, vis_params, 'SRTM DEM', True, 1) 26 | m.addLayerControl() 27 | 28 | # call to render Folium map in Streamlit 29 | folium_static(m) 30 | -------------------------------------------------------------------------------- /requirements.txt: -------------------------------------------------------------------------------- 1 | streamlit 2 | streamlit-folium 3 | geemap 4 | leafmap 5 | click==8 -------------------------------------------------------------------------------- /setup.sh: -------------------------------------------------------------------------------- 1 | 2 | mkdir -p ~/.streamlit/ 3 | echo "[general] 4 | email = \"giswqs@gmail.com\" 5 | " > ~/.streamlit/credentials.toml 6 | echo "[server] 7 | headless = true 8 | port = $PORT 9 | enableCORS = false 10 | " > ~/.streamlit/config.toml -------------------------------------------------------------------------------- /streamlit_app.py: -------------------------------------------------------------------------------- 1 | import streamlit as st 2 | from streamlit_folium import folium_static 3 | import folium 4 | 5 | "# streamlit-folium" 6 | 7 | with st.echo(): 8 | import streamlit as st 9 | from streamlit_folium import folium_static 10 | import folium 11 | 12 | # center on Liberty Bell 13 | m = folium.Map(location=[39.949610, -75.150282], zoom_start=16) 14 | 15 | # add marker for Liberty Bell 16 | tooltip = "Liberty Bell" 17 | folium.Marker( 18 | [39.949610, -75.150282], popup="Liberty Bell", tooltip=tooltip 19 | ).add_to(m) 20 | 21 | # call to render Folium map in Streamlit 22 | folium_static(m) 23 | --------------------------------------------------------------------------------