yusufenes commited on
Commit
36d5083
·
verified ·
1 Parent(s): 8c32736

Update app.py

Browse files
Files changed (1) hide show
  1. app.py +262 -240
app.py CHANGED
@@ -1,240 +1,262 @@
1
- import streamlit as st
2
- import joblib
3
- import pandas as pd
4
- import numpy as np
5
- from preprocess_pipeline import SingleInstancePreprocessor
6
- from get_real_home_listing import get_home_listings
7
- from home_listing_display import display_home_listings
8
-
9
- model = joblib.load('random_forest.joblib')
10
- preprocessor = SingleInstancePreprocessor()
11
-
12
-
13
- temp = {}
14
- st.set_page_config(
15
- page_title="🏠 Ev Fiyatı Tahmin ve Öneri aracı",
16
- page_icon="🏠",
17
- layout="wide",
18
- initial_sidebar_state="expanded",
19
- )
20
-
21
- st.markdown("""
22
- <style>
23
- :root {
24
- --primary: #2A3950;
25
- --secondary: #00C0A3;
26
- --background: #1A1A1A;
27
- --surface: #2D2D2D;
28
- --text-primary: #FFFFFF;
29
- }
30
-
31
- body {
32
- font-family: 'Arial', sans-serif;
33
- background-color: var(--background);
34
- color: var(--text-primary);
35
- }
36
-
37
- .stApp {
38
- background: var(--background);
39
- color: var(--text-primary);
40
- font-family: 'Inter', sans-serif;
41
- }
42
-
43
- .header-container {
44
- background: var(--surface);
45
- padding: 3rem;
46
- border-radius: 15px;
47
- margin-bottom: 2.5rem;
48
- box-shadow: 0 6px 25px rgba(0,0,0,0.3);
49
- text-align: center;
50
- }
51
-
52
- .input-card {
53
- background: var(--surface);
54
- border-radius: 12px;
55
- padding: 2rem;
56
- margin: 1.2rem 0;
57
- border: 1px solid rgba(255,255,255,0.1);
58
- transition: all 0.3s ease;
59
- }
60
-
61
- .input-card:hover {
62
- transform: translateY(-5px);
63
- box-shadow: 0 8px 20px rgba(0,0,0,0.4);
64
- }
65
-
66
- .stSelectbox, .stNumberInput, .stTextInput {
67
- background: var(--primary) !important;
68
- border-radius: 10px !important;
69
- border: none !important;
70
- color: var(--text-primary) !important;
71
- padding: 0.75rem !important;
72
- }
73
-
74
- .stButton>button {
75
- background: var(--secondary) !important;
76
- color: var(--text-primary) !important;
77
- border: none;
78
- padding: 16px 36px;
79
- border-radius: 10px;
80
- font-weight: 600;
81
- transition: all 0.3s;
82
- box-shadow: 0 5px 15px rgba(0,192,163,0.4);
83
- }
84
-
85
- .stButton>button:hover {
86
- opacity: 0.9;
87
- transform: translateY(-2px);
88
- box-shadow: 0 6px 20px rgba(0,192,163,0.5);
89
- }
90
-
91
- .prediction-card {
92
- background: linear-gradient(135deg, var(--primary) 0%, #1B2838 100%);
93
- border-radius: 15px;
94
- padding: 2.5rem;
95
- text-align: center;
96
- margin: 2.5rem 0;
97
- border: 1px solid rgba(255,255,255,0.15);
98
- box-shadow: 0 8px 25px rgba(0,0,0,0.3);
99
- }
100
-
101
- .section-title {
102
- font-size: 1.4rem;
103
- font-weight: 700;
104
- color: var(--secondary);
105
- margin: 1.8rem 0;
106
- display: flex;
107
- align-items: center;
108
- gap: 1rem;
109
- text-transform: uppercase;
110
- letter-spacing: 1px;
111
- }
112
-
113
- .footer {
114
- text-align: center;
115
- padding: 2rem;
116
- color: rgba(255,255,255,0.7);
117
- font-size: 0.95rem;
118
- margin-top: 3.5rem;
119
- border-top: 1px solid rgba(255,255,255,0.1);
120
- }
121
-
122
- /* Ek olarak, mobil uyumluluk için medya sorguları ekleyebilirsiniz */
123
- @media (max-width: 768px) {
124
- .header-container {
125
- padding: 2rem;
126
- margin-bottom: 2rem;
127
- }
128
- .section-title {
129
- font-size: 1.2rem;
130
- margin: 1.5rem 0;
131
- }
132
- }
133
- </style>
134
- """, unsafe_allow_html=True)
135
-
136
- with st.container():
137
- st.markdown("""
138
- <div class="header-container">
139
- <h1 style="margin:0;font-size:2.5rem">🏠 Ev Tahmin</h1>
140
- <p style="color:rgba(255,255,255,0.8);margin-top:0.5rem">Ev Fiyatı Tahmin Ve Öneri Aracı</p>
141
- </div>
142
- """, unsafe_allow_html=True)
143
-
144
- with st.container():
145
- col1, col2 = st.columns([1, 1.2])
146
-
147
- with col1:
148
- st.markdown('<div class="section-title">📍 Lokasyon Bilgileri</div>', unsafe_allow_html=True)
149
- st.markdown('<div style="color:rgba(255,255,255,0.8);margin-bottom:1rem">Ev fiyatı tahmini yapmak için lütfen aşağıdaki bilgileri doldurun.</div>', unsafe_allow_html=True)
150
-
151
- with st.container():
152
- st.markdown('<div class="input-card">', unsafe_allow_html=True)
153
- il_listesi = list(preprocessor.location_map.keys())
154
- selected_il = st.selectbox('İl', options=il_listesi)
155
- ilceler = list(preprocessor.location_map.get(selected_il, {}).keys())
156
- selected_ilce = st.selectbox('İlçe', options=ilceler)
157
- mahalleler = preprocessor.location_map.get(selected_il, {}).get(selected_ilce, [])
158
- selected_mahalle = st.selectbox('Mahalle', options=mahalleler)
159
- st.markdown('</div>', unsafe_allow_html=True)
160
-
161
- st.markdown('<div class="section-title">📐 Temel Özellikler</div>', unsafe_allow_html=True)
162
- with st.container():
163
- st.markdown('<div class="input-card">', unsafe_allow_html=True)
164
- tipi = st.selectbox('Konut Tipi', options=['Daire', 'Residence', 'Villa', 'Müstakil Ev', 'Yazlık'])
165
- brut_metrekare = st.number_input('Brüt Metrekare (m²)', min_value=30, max_value=500, value=100)
166
- binanin_yasi = st.number_input('Bina Yaşı', min_value=0, max_value=100, value=5)
167
- st.markdown('</div>', unsafe_allow_html=True)
168
-
169
- with col2:
170
- st.markdown('<div class="section-title">📊 Detaylı Özellikler</div>', unsafe_allow_html=True)
171
- with st.container():
172
- st.markdown('<div class="input-card">', unsafe_allow_html=True)
173
- cols = st.columns(2)
174
- with cols[0]:
175
- oda_sayisi = st.selectbox('Oda', options=['1+1', '2+1', '3+1', '4+1', '5+1'])
176
- banyo_sayisi = st.number_input('Banyo', min_value=1, max_value=5, value=1)
177
- isitma_tipi = st.selectbox('Isıtma Tipi', options=['Doğalgaz', 'Merkezi Sistem', 'Elektrikli', 'Soba'])
178
- with cols[1]:
179
- bulundugu_kat = 3 # Ortalama veya tipik bir değer
180
- binanin_kat_sayisi = 10 # Ortalama veya tipik bir değer
181
- esya_durumu = st.selectbox('Eşya Durumu', options=['Eşyalı', 'Eşyasız'])
182
- st.markdown('</div>', unsafe_allow_html=True)
183
-
184
- st.markdown('<div class="section-title">⚙️ Diğer Ayarlar</div>', unsafe_allow_html=True)
185
- with st.container():
186
- st.markdown('<div class="input-card">', unsafe_allow_html=True)
187
- cols = st.columns(2)
188
- with cols[0]:
189
- site_icerisinde = st.selectbox('Site İçinde', options=['Evet', 'Hayır'])
190
- with cols[1]:
191
- kullanim_durumu = st.selectbox('Kullanım Durumu', options=['Konut', 'İş Yeri', 'Diğer'])
192
- st.markdown('</div>', unsafe_allow_html=True)
193
-
194
- if st.button('💵 Tahmin Yap', use_container_width=True):
195
- input_data = {
196
- 'Oda Sayısı': oda_sayisi,
197
- 'Bulunduğu Kat': bulundugu_kat,
198
- 'Isıtma Tipi': isitma_tipi,
199
- 'Eşya Durumu': esya_durumu,
200
- 'Site İçerisinde': site_icerisinde,
201
- 'Tipi': tipi,
202
- 'Brüt Metrekare': brut_metrekare,
203
- 'Binanın Yaşı': binanin_yasi,
204
- 'Binanın Kat Sayısı': binanin_kat_sayisi,
205
- 'Kullanım Durumu': kullanim_durumu,
206
- 'Banyo Sayısı': banyo_sayisi,
207
- 'İl': selected_il,
208
- 'İlçe': selected_ilce,
209
- 'Mahalle': selected_mahalle
210
- }
211
-
212
-
213
- with st.spinner('AI analiz yapıyor...'):
214
- input_array = preprocessor.transform(input_data)
215
-
216
- tahmin = model.predict(input_array)
217
-
218
- st.markdown(f'''
219
- <div class="prediction-card">
220
- <h3 style="margin:0 0 1rem 0;color:rgba(255,255,255,0.9)">TAHMİN EDİLEN DEĞER</h3>
221
- <div style="font-size:2.8rem;font-weight:700;color:var(--secondary);margin:1rem 0">{tahmin[0]:,.0f} TL</div>
222
- <p style="color:rgba(255,255,255,0.7);margin:0">*Gerçek değerler piyasa koşullarına göre değişiklik gösterebilir</p>
223
- </div>
224
- ''', unsafe_allow_html=True)
225
-
226
- predicted_price = tahmin[0]
227
- with st.spinner("Seçim ile benzer özelliğe sahip ilanlar getiriliyor..."):
228
- home_listings = get_home_listings(selected_il, predicted_price)
229
- display_home_listings(home_listings)
230
-
231
- st.markdown('''
232
- <div class="footer">
233
- <div> @yusufenes || Konut Fiyatı Tahmin ve Konut Öneri Sistemi || Tüm veriler İnternet Üzerinden Alınmıştır Ticari Kullanım Hakkı Yoktur.</div>
234
- <div style="margin-top:0.5rem;font-size:0.8rem">
235
- <span style="opacity:0.7">Geliştirici: Yusuf Enes</span> •
236
- <span style="opacity:0.7">🧑‍💻 <a href="https://github.com/yusuffenes">Github</a> </span>
237
- </div>
238
- </div>
239
- ''', unsafe_allow_html=True)
240
-
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import streamlit as st
2
+ import joblib
3
+ import pandas as pd
4
+ import numpy as np
5
+ from preprocess_pipeline import SingleInstancePreprocessor
6
+ from get_real_home_listing import get_home_listings
7
+ from home_listing_display import display_home_listings
8
+
9
+ model = joblib.load('random_forest.joblib')
10
+ preprocessor = SingleInstancePreprocessor()
11
+
12
+
13
+ def install_chrome():
14
+ if not os.path.exists("/usr/bin/google-chrome"):
15
+ print("Installing Google Chrome...")
16
+ subprocess.run(
17
+ "wget -q -O - https://dl-ssl.google.com/linux/linux_signing_key.pub | sudo apt-key add - && "
18
+ "echo 'deb [arch=amd64] http://dl.google.com/linux/chrome/deb/ stable main' | sudo tee /etc/apt/sources.list.d/google-chrome.list && "
19
+ "sudo apt update && sudo apt install -y google-chrome-stable",
20
+ shell=True,
21
+ check=True,
22
+ )
23
+
24
+ def install_chromedriver():
25
+ if not os.path.exists("/usr/local/bin/chromedriver"):
26
+ print("Installing ChromeDriver...")
27
+ subprocess.run(
28
+ "wget -O /usr/local/bin/chromedriver https://storage.googleapis.com/chrome-for-testing-public/133.0.6943.98/linux64/chromedriver-linux64.zip && "
29
+ "unzip /usr/local/bin/chromedriver -d /usr/local/bin/ && "
30
+ "chmod +x /usr/local/bin/chromedriver",
31
+ shell=True,
32
+ check=True,
33
+ )
34
+
35
+ temp = {}
36
+ st.set_page_config(
37
+ page_title="🏠 Ev Fiyatı Tahmin ve Öneri aracı",
38
+ page_icon="🏠",
39
+ layout="wide",
40
+ initial_sidebar_state="expanded",
41
+ )
42
+
43
+ st.markdown("""
44
+ <style>
45
+ :root {
46
+ --primary: #2A3950;
47
+ --secondary: #00C0A3;
48
+ --background: #1A1A1A;
49
+ --surface: #2D2D2D;
50
+ --text-primary: #FFFFFF;
51
+ }
52
+
53
+ body {
54
+ font-family: 'Arial', sans-serif;
55
+ background-color: var(--background);
56
+ color: var(--text-primary);
57
+ }
58
+
59
+ .stApp {
60
+ background: var(--background);
61
+ color: var(--text-primary);
62
+ font-family: 'Inter', sans-serif;
63
+ }
64
+
65
+ .header-container {
66
+ background: var(--surface);
67
+ padding: 3rem;
68
+ border-radius: 15px;
69
+ margin-bottom: 2.5rem;
70
+ box-shadow: 0 6px 25px rgba(0,0,0,0.3);
71
+ text-align: center;
72
+ }
73
+
74
+ .input-card {
75
+ background: var(--surface);
76
+ border-radius: 12px;
77
+ padding: 2rem;
78
+ margin: 1.2rem 0;
79
+ border: 1px solid rgba(255,255,255,0.1);
80
+ transition: all 0.3s ease;
81
+ }
82
+
83
+ .input-card:hover {
84
+ transform: translateY(-5px);
85
+ box-shadow: 0 8px 20px rgba(0,0,0,0.4);
86
+ }
87
+
88
+ .stSelectbox, .stNumberInput, .stTextInput {
89
+ background: var(--primary) !important;
90
+ border-radius: 10px !important;
91
+ border: none !important;
92
+ color: var(--text-primary) !important;
93
+ padding: 0.75rem !important;
94
+ }
95
+
96
+ .stButton>button {
97
+ background: var(--secondary) !important;
98
+ color: var(--text-primary) !important;
99
+ border: none;
100
+ padding: 16px 36px;
101
+ border-radius: 10px;
102
+ font-weight: 600;
103
+ transition: all 0.3s;
104
+ box-shadow: 0 5px 15px rgba(0,192,163,0.4);
105
+ }
106
+
107
+ .stButton>button:hover {
108
+ opacity: 0.9;
109
+ transform: translateY(-2px);
110
+ box-shadow: 0 6px 20px rgba(0,192,163,0.5);
111
+ }
112
+
113
+ .prediction-card {
114
+ background: linear-gradient(135deg, var(--primary) 0%, #1B2838 100%);
115
+ border-radius: 15px;
116
+ padding: 2.5rem;
117
+ text-align: center;
118
+ margin: 2.5rem 0;
119
+ border: 1px solid rgba(255,255,255,0.15);
120
+ box-shadow: 0 8px 25px rgba(0,0,0,0.3);
121
+ }
122
+
123
+ .section-title {
124
+ font-size: 1.4rem;
125
+ font-weight: 700;
126
+ color: var(--secondary);
127
+ margin: 1.8rem 0;
128
+ display: flex;
129
+ align-items: center;
130
+ gap: 1rem;
131
+ text-transform: uppercase;
132
+ letter-spacing: 1px;
133
+ }
134
+
135
+ .footer {
136
+ text-align: center;
137
+ padding: 2rem;
138
+ color: rgba(255,255,255,0.7);
139
+ font-size: 0.95rem;
140
+ margin-top: 3.5rem;
141
+ border-top: 1px solid rgba(255,255,255,0.1);
142
+ }
143
+
144
+ /* Ek olarak, mobil uyumluluk için medya sorguları ekleyebilirsiniz */
145
+ @media (max-width: 768px) {
146
+ .header-container {
147
+ padding: 2rem;
148
+ margin-bottom: 2rem;
149
+ }
150
+ .section-title {
151
+ font-size: 1.2rem;
152
+ margin: 1.5rem 0;
153
+ }
154
+ }
155
+ </style>
156
+ """, unsafe_allow_html=True)
157
+
158
+ with st.container():
159
+ st.markdown("""
160
+ <div class="header-container">
161
+ <h1 style="margin:0;font-size:2.5rem">🏠 Ev Tahmin</h1>
162
+ <p style="color:rgba(255,255,255,0.8);margin-top:0.5rem">Ev Fiyatı Tahmin Ve Öneri Aracı</p>
163
+ </div>
164
+ """, unsafe_allow_html=True)
165
+
166
+ with st.container():
167
+ col1, col2 = st.columns([1, 1.2])
168
+
169
+ with col1:
170
+ st.markdown('<div class="section-title">📍 Lokasyon Bilgileri</div>', unsafe_allow_html=True)
171
+ st.markdown('<div style="color:rgba(255,255,255,0.8);margin-bottom:1rem">Ev fiyatı tahmini yapmak için lütfen aşağıdaki bilgileri doldurun.</div>', unsafe_allow_html=True)
172
+
173
+ with st.container():
174
+ st.markdown('<div class="input-card">', unsafe_allow_html=True)
175
+ il_listesi = list(preprocessor.location_map.keys())
176
+ selected_il = st.selectbox('İl', options=il_listesi)
177
+ ilceler = list(preprocessor.location_map.get(selected_il, {}).keys())
178
+ selected_ilce = st.selectbox('İlçe', options=ilceler)
179
+ mahalleler = preprocessor.location_map.get(selected_il, {}).get(selected_ilce, [])
180
+ selected_mahalle = st.selectbox('Mahalle', options=mahalleler)
181
+ st.markdown('</div>', unsafe_allow_html=True)
182
+
183
+ st.markdown('<div class="section-title">📐 Temel Özellikler</div>', unsafe_allow_html=True)
184
+ with st.container():
185
+ st.markdown('<div class="input-card">', unsafe_allow_html=True)
186
+ tipi = st.selectbox('Konut Tipi', options=['Daire', 'Residence', 'Villa', 'Müstakil Ev', 'Yazlık'])
187
+ brut_metrekare = st.number_input('Brüt Metrekare (m²)', min_value=30, max_value=500, value=100)
188
+ binanin_yasi = st.number_input('Bina Yaşı', min_value=0, max_value=100, value=5)
189
+ st.markdown('</div>', unsafe_allow_html=True)
190
+
191
+ with col2:
192
+ st.markdown('<div class="section-title">📊 Detaylı Özellikler</div>', unsafe_allow_html=True)
193
+ with st.container():
194
+ st.markdown('<div class="input-card">', unsafe_allow_html=True)
195
+ cols = st.columns(2)
196
+ with cols[0]:
197
+ oda_sayisi = st.selectbox('Oda', options=['1+1', '2+1', '3+1', '4+1', '5+1'])
198
+ banyo_sayisi = st.number_input('Banyo', min_value=1, max_value=5, value=1)
199
+ isitma_tipi = st.selectbox('Isıtma Tipi', options=['Doğalgaz', 'Merkezi Sistem', 'Elektrikli', 'Soba'])
200
+ with cols[1]:
201
+ bulundugu_kat = 3 # Ortalama veya tipik bir değer
202
+ binanin_kat_sayisi = 10 # Ortalama veya tipik bir değer
203
+ esya_durumu = st.selectbox('Eşya Durumu', options=['Eşyalı', 'Eşyasız'])
204
+ st.markdown('</div>', unsafe_allow_html=True)
205
+
206
+ st.markdown('<div class="section-title">⚙️ Diğer Ayarlar</div>', unsafe_allow_html=True)
207
+ with st.container():
208
+ st.markdown('<div class="input-card">', unsafe_allow_html=True)
209
+ cols = st.columns(2)
210
+ with cols[0]:
211
+ site_icerisinde = st.selectbox('Site İçinde', options=['Evet', 'Hayır'])
212
+ with cols[1]:
213
+ kullanim_durumu = st.selectbox('Kullanım Durumu', options=['Konut', 'İş Yeri', 'Diğer'])
214
+ st.markdown('</div>', unsafe_allow_html=True)
215
+
216
+ if st.button('💵 Tahmin Yap', use_container_width=True):
217
+ input_data = {
218
+ 'Oda Sayısı': oda_sayisi,
219
+ 'Bulunduğu Kat': bulundugu_kat,
220
+ 'Isıtma Tipi': isitma_tipi,
221
+ 'Eşya Durumu': esya_durumu,
222
+ 'Site İçerisinde': site_icerisinde,
223
+ 'Tipi': tipi,
224
+ 'Brüt Metrekare': brut_metrekare,
225
+ 'Binanın Yaşı': binanin_yasi,
226
+ 'Binanın Kat Sayısı': binanin_kat_sayisi,
227
+ 'Kullanım Durumu': kullanim_durumu,
228
+ 'Banyo Sayısı': banyo_sayisi,
229
+ 'İl': selected_il,
230
+ 'İlçe': selected_ilce,
231
+ 'Mahalle': selected_mahalle
232
+ }
233
+
234
+
235
+ with st.spinner('AI analiz yapıyor...'):
236
+ input_array = preprocessor.transform(input_data)
237
+
238
+ tahmin = model.predict(input_array)
239
+
240
+ st.markdown(f'''
241
+ <div class="prediction-card">
242
+ <h3 style="margin:0 0 1rem 0;color:rgba(255,255,255,0.9)">TAHMİN EDİLEN DEĞER</h3>
243
+ <div style="font-size:2.8rem;font-weight:700;color:var(--secondary);margin:1rem 0">{tahmin[0]:,.0f} TL</div>
244
+ <p style="color:rgba(255,255,255,0.7);margin:0">*Gerçek değerler piyasa koşullarına göre değişiklik gösterebilir</p>
245
+ </div>
246
+ ''', unsafe_allow_html=True)
247
+
248
+ predicted_price = tahmin[0]
249
+ with st.spinner("Seçim ile benzer özelliğe sahip ilanlar getiriliyor..."):
250
+ home_listings = get_home_listings(selected_il, predicted_price)
251
+ display_home_listings(home_listings)
252
+
253
+ st.markdown('''
254
+ <div class="footer">
255
+ <div> @yusufenes || Konut Fiyatı Tahmin ve Konut Öneri Sistemi || Tüm veriler İnternet Üzerinden Alınmıştır Ticari Kullanım Hakkı Yoktur.</div>
256
+ <div style="margin-top:0.5rem;font-size:0.8rem">
257
+ <span style="opacity:0.7">Geliştirici: Yusuf Enes</span> •
258
+ <span style="opacity:0.7">🧑‍💻 <a href="https://github.com/yusuffenes">Github</a> </span>
259
+ </div>
260
+ </div>
261
+ ''', unsafe_allow_html=True)
262
+