shikharyashmaurya commited on
Commit
d173cda
·
verified ·
1 Parent(s): fb83ace

Upload user-authenticated-revision-app5.py

Browse files
Files changed (1) hide show
  1. user-authenticated-revision-app5.py +259 -0
user-authenticated-revision-app5.py ADDED
@@ -0,0 +1,259 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import streamlit as st
2
+ import networkx as nx
3
+ import matplotlib.pyplot as plt
4
+ import textwrap
5
+ from io import BytesIO
6
+ import uuid
7
+
8
+ class RevisionApp:
9
+ def __init__(self):
10
+ self.initialize_session_state()
11
+ self.handle_user_selection()
12
+ self.create_ui()
13
+
14
+ def initialize_session_state(self):
15
+ if 'users' not in st.session_state:
16
+ st.session_state.users = {}
17
+ if 'current_user' not in st.session_state:
18
+ st.session_state.current_user = None
19
+ if 'user_sessions' not in st.session_state:
20
+ st.session_state.user_sessions = {}
21
+
22
+ def handle_user_selection(self):
23
+ st.sidebar.title("User Selection")
24
+ user_name = st.sidebar.text_input("Enter your name:")
25
+ if st.sidebar.button("Set User"):
26
+ if user_name:
27
+ # Check if the username is already in use in this session
28
+ if user_name in st.session_state.user_sessions:
29
+ st.sidebar.error(f"The name '{user_name}' is already in use. Please choose a different name.")
30
+ else:
31
+ # Generate a unique session ID for this user
32
+ session_id = str(uuid.uuid4())
33
+ st.session_state.current_user = f"{user_name}_{session_id}"
34
+ st.session_state.user_sessions[user_name] = session_id
35
+ if st.session_state.current_user not in st.session_state.users:
36
+ st.session_state.users[st.session_state.current_user] = {}
37
+ st.success(f"User set to: {user_name}")
38
+ st.experimental_rerun()
39
+
40
+ def create_ui(self):
41
+ if st.session_state.current_user:
42
+ display_name = st.session_state.current_user.split('_')[0] # Extract the user's name without the session ID
43
+ st.title(f"Concept Revision App - Welcome, {display_name}!")
44
+
45
+ # Navigation
46
+ page = st.sidebar.selectbox("Choose a page", ["Tree View", "Mind Map", "Search"])
47
+
48
+ if page == "Tree View":
49
+ self.show_tree_view()
50
+ elif page == "Mind Map":
51
+ self.show_mind_map()
52
+ elif page == "Search":
53
+ self.show_search()
54
+ else:
55
+ st.title("Concept Revision App")
56
+ st.write("Please enter your name in the sidebar to begin.")
57
+
58
+ # The rest of the methods (show_tree_view, show_concept_details, show_mind_map, show_search, search_data, custom_tree_layout)
59
+ # remain the same as in the previous version, just ensure you're using st.session_state.current_user
60
+ # to access the correct user data in st.session_state.users
61
+
62
+ # Example of how to modify a method to use the new user identifier:
63
+ def show_tree_view(self):
64
+ st.header("Tree View")
65
+
66
+ # Input for new concept
67
+ new_key = st.text_input("Enter a new concept:")
68
+ if st.button("Add Concept"):
69
+ if new_key and new_key not in st.session_state.users[st.session_state.current_user]:
70
+ st.session_state.users[st.session_state.current_user][new_key] = {'next': [], 'text': []}
71
+ st.success(f"Added new concept: {new_key}")
72
+ st.experimental_rerun()
73
+
74
+ # Display concepts
75
+ user_data = st.session_state.users[st.session_state.current_user]
76
+ selected_concept = st.selectbox("Select a concept to view details:",
77
+ options=[""] + list(user_data.keys()))
78
+
79
+ if selected_concept:
80
+ self.show_concept_details(selected_concept)
81
+
82
+ # ... (other methods remain the same, just ensure you're using st.session_state.current_user consistently)
83
+ # def show_tree_view(self):
84
+ # st.header("Tree View")
85
+
86
+ # # Input for new concept
87
+ # new_key = st.text_input("Enter a new concept:")
88
+ # if st.button("Add Concept"):
89
+ # if new_key and new_key not in st.session_state.users[st.session_state.current_user]:
90
+ # st.session_state.users[st.session_state.current_user][new_key] = {'next': [], 'text': []}
91
+ # st.success(f"Added new concept: {new_key}")
92
+ # st.experimental_rerun()
93
+
94
+ # # Display concepts
95
+ # user_data = st.session_state.users[st.session_state.current_user]
96
+ # selected_concept = st.selectbox("Select a concept to view details:",
97
+ # options=[""] + list(user_data.keys()))
98
+
99
+ # if selected_concept:
100
+ # self.show_concept_details(selected_concept)
101
+
102
+ def show_concept_details(self, key):
103
+ user_data = st.session_state.users[st.session_state.current_user]
104
+ st.subheader(f"Concept: {key}")
105
+
106
+ # Display related concepts
107
+ st.write("Related Concepts:")
108
+ for next_item in user_data[key]['next']:
109
+ if st.button(f"Go to {next_item}", key=f"goto_{next_item}"):
110
+ self.show_concept_details(next_item)
111
+ return
112
+
113
+ # Add related concept
114
+ new_related = st.text_input(f"Add related concept to {key}:", key=f"related_{key}")
115
+ if st.button(f"Add related to {key}", key=f"add_related_{key}"):
116
+ if new_related and new_related not in user_data[key]['next']:
117
+ if new_related not in user_data:
118
+ user_data[new_related] = {'next': [], 'text': []}
119
+ user_data[key]['next'].append(new_related)
120
+ st.success(f"Added {new_related} as related to {key}")
121
+ st.experimental_rerun()
122
+
123
+ # Display information
124
+ st.write("Information:")
125
+ for i, text_item in enumerate(user_data[key]['text']):
126
+ st.text_area(f"Info {i+1}", value=text_item, key=f"info_{key}_{i}", height=100, disabled=True)
127
+
128
+ # Add information
129
+ new_info = st.text_area(f"Add information to {key}:", key=f"new_info_{key}")
130
+ if st.button(f"Add info to {key}", key=f"add_info_{key}"):
131
+ if new_info:
132
+ user_data[key]['text'].append(new_info)
133
+ st.success(f"Added new information to {key}")
134
+ st.experimental_rerun()
135
+
136
+ def show_mind_map(self):
137
+ st.header("Mind Map")
138
+
139
+ user_data = st.session_state.users[st.session_state.current_user]
140
+ G = nx.Graph()
141
+ for key, value in user_data.items():
142
+ G.add_node(key)
143
+ for next_item in value['next']:
144
+ if next_item in user_data:
145
+ G.add_edge(key, next_item)
146
+
147
+ pos = self.custom_tree_layout(G)
148
+
149
+ plt.figure(figsize=(12, 8))
150
+ nx.draw(G, pos, with_labels=False, node_color='lightblue', node_size=3000, alpha=0.8)
151
+
152
+ for node, (x, y) in pos.items():
153
+ lines = textwrap.wrap(node, width=10)
154
+ plt.annotate('\n'.join(lines), (x, y), horizontalalignment='center', verticalalignment='center')
155
+
156
+ plt.axis('off')
157
+
158
+ buf = BytesIO()
159
+ plt.savefig(buf, format="png")
160
+ buf.seek(0)
161
+
162
+ st.image(buf, caption='Mind Map', use_column_width=True)
163
+
164
+ def show_search(self):
165
+ st.header("Search")
166
+
167
+ query = st.text_input("Enter search term:")
168
+ if st.button("Search"):
169
+ results = self.search_data(query)
170
+ if results:
171
+ for key in results:
172
+ with st.expander(f"Concept: {key}"):
173
+ st.write("Related Concepts:")
174
+ for related in st.session_state.users[st.session_state.current_user][key]['next']:
175
+ st.write(f"- {related}")
176
+ st.write("Information:")
177
+ for info in st.session_state.users[st.session_state.current_user][key]['text']:
178
+ st.write(f"- {info}")
179
+ else:
180
+ st.write("No results found.")
181
+
182
+ def search_data(self, query):
183
+ query = query.lower()
184
+ user_data = st.session_state.users[st.session_state.current_user]
185
+ results = set()
186
+ for key, value in user_data.items():
187
+ if query in key.lower():
188
+ results.add(key)
189
+ for next_item in value['next']:
190
+ if query in next_item.lower():
191
+ results.add(key)
192
+ for text_item in value['text']:
193
+ if query in text_item.lower():
194
+ results.add(key)
195
+ return list(results)
196
+
197
+ def custom_tree_layout(self, G):
198
+ if not G.nodes():
199
+ return {}
200
+
201
+ def bfs_tree(root):
202
+ tree = nx.bfs_tree(G, root)
203
+ return tree
204
+
205
+ def assign_positions(tree, root):
206
+ pos = {}
207
+ level_width = {}
208
+ max_depth = 0
209
+
210
+ def dfs(node, depth, order):
211
+ nonlocal max_depth
212
+ max_depth = max(max_depth, depth)
213
+ if depth not in level_width:
214
+ level_width[depth] = 0
215
+ level_width[depth] += 1
216
+ children = list(tree.successors(node))
217
+ if not children:
218
+ pos[node] = (order, -depth)
219
+ return order + 1
220
+
221
+ start = order
222
+ for child in children:
223
+ order = dfs(child, depth + 1, order)
224
+ pos[node] = (start + (order - start - 1) / 2, -depth)
225
+ return order
226
+
227
+ dfs(root, 0, 0)
228
+
229
+ # Normalize positions
230
+ max_width = max(level_width.values()) if level_width else 1
231
+ for node in pos:
232
+ x, y = pos[node]
233
+ pos[node] = (x / max_width, y / max_depth if max_depth != 0 else 0)
234
+
235
+ return pos
236
+
237
+ # Handle disconnected components
238
+ components = list(nx.connected_components(G))
239
+ if not components:
240
+ return {}
241
+
242
+ pos = {}
243
+ y_offset = 0
244
+ for component in components:
245
+ subgraph = G.subgraph(component)
246
+ root = max(subgraph.nodes(), key=lambda n: subgraph.degree(n))
247
+ tree = bfs_tree(root)
248
+ component_pos = assign_positions(tree, root)
249
+
250
+ # Adjust y-positions for each component
251
+ for node, (x, y) in component_pos.items():
252
+ pos[node] = (x, y + y_offset)
253
+
254
+ y_offset -= 1.5 # Increase vertical separation between components
255
+
256
+ return pos
257
+
258
+ if __name__ == "__main__":
259
+ app = RevisionApp()